aboutsummaryrefslogtreecommitdiff
path: root/Ryujinx.Graphics.Nvdec.FFmpeg/Native/FFmpegApi.cs
blob: 9bf71778bd79896b0a92c9dfce8191a9843b4ea8 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
using System;
using System.Collections.Generic;
using System.Reflection;
using System.Runtime.InteropServices;

namespace Ryujinx.Graphics.Nvdec.FFmpeg.Native
{
    static class FFmpegApi
    {
        public const string AvCodecLibraryName = "avcodec";
        public const string AvUtilLibraryName = "avutil";

        private static readonly Dictionary<string, (int, int)> _librariesWhitelist = new Dictionary<string, (int, int)>
        {
            { AvCodecLibraryName, (58, 59) },
            { AvUtilLibraryName, (56, 57) }
        };

        private static string FormatLibraryNameForCurrentOs(string libraryName, int version)
        {
            if (OperatingSystem.IsWindows())
            {
                return $"{libraryName}-{version}.dll";
            }
            else if (OperatingSystem.IsLinux())
            {
                return $"lib{libraryName}.so.{version}";
            }
            else if (OperatingSystem.IsMacOS())
            {
                return $"lib{libraryName}.{version}.dylib";
            }
            else
            {
                throw new NotImplementedException($"Unsupported OS for FFmpeg: {RuntimeInformation.RuntimeIdentifier}");
            }
        }


        private static bool TryLoadWhitelistedLibrary(string libraryName, Assembly assembly, DllImportSearchPath? searchPath, out IntPtr handle)
        {
            handle = IntPtr.Zero;

            if (_librariesWhitelist.TryGetValue(libraryName, out var value))
            {
                (int minVersion, int maxVersion) = value;

                for (int version = maxVersion; version >= minVersion; version--)
                {
                    if (NativeLibrary.TryLoad(FormatLibraryNameForCurrentOs(libraryName, version), assembly, searchPath, out handle))
                    {
                        return true;
                    }
                }
            }

            return false;
        }

        static FFmpegApi()
        {
            NativeLibrary.SetDllImportResolver(typeof(FFmpegApi).Assembly, (name, assembly, path) =>
            {
                IntPtr handle;

                if (name == AvUtilLibraryName && TryLoadWhitelistedLibrary(AvUtilLibraryName, assembly, path, out handle))
                {
                    return handle;
                }
                else if (name == AvCodecLibraryName && TryLoadWhitelistedLibrary(AvCodecLibraryName, assembly, path, out handle))
                {
                    return handle;
                }

                return IntPtr.Zero;
            });
        }

        public unsafe delegate void av_log_set_callback_callback(void* a0, AVLog level, [MarshalAs(UnmanagedType.LPUTF8Str)] string a2, byte* a3);

        [DllImport(AvUtilLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern AVFrame* av_frame_alloc();

        [DllImport(AvUtilLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern void av_frame_unref(AVFrame* frame);

        [DllImport(AvUtilLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern void av_free(AVFrame* frame);

        [DllImport(AvUtilLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern void av_log_set_level(AVLog level);

        [DllImport(AvUtilLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern void av_log_set_callback(av_log_set_callback_callback callback);

        [DllImport(AvUtilLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern AVLog av_log_get_level();

        [DllImport(AvUtilLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern void av_log_format_line(void* ptr, AVLog level, [MarshalAs(UnmanagedType.LPUTF8Str)] string fmt, byte* vl, byte* line, int lineSize, int* printPrefix);

        [DllImport(AvCodecLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern AVCodec* avcodec_find_decoder(AVCodecID id);

        [DllImport(AvCodecLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern AVCodecContext* avcodec_alloc_context3(AVCodec* codec);

        [DllImport(AvCodecLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern int avcodec_open2(AVCodecContext* avctx, AVCodec* codec, void **options);

        [DllImport(AvCodecLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern int avcodec_close(AVCodecContext* avctx);

        [DllImport(AvCodecLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern void avcodec_free_context(AVCodecContext** avctx);

        [DllImport(AvCodecLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern AVPacket* av_packet_alloc();

        [DllImport(AvCodecLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern void av_packet_unref(AVPacket* pkt);

        [DllImport(AvCodecLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern void av_packet_free(AVPacket** pkt);

        [DllImport(AvCodecLibraryName, CallingConvention = CallingConvention.Cdecl)]
        internal static unsafe extern int avcodec_version();
    }
}