From fbb38d61fdeb6f48b42979f1d1f2786268e329e9 Mon Sep 17 00:00:00 2001 From: YoshiRulz Date: Tue, 28 Jul 2020 20:24:54 +1000 Subject: [PATCH] Use a DLIR for Mupen core and plugins AttachPlugin still returns IntPtr (thanks to reflection hack) so the code for audio/input/video is unaffected --- .../N64/NativeApi/mupen64plusCoreApi.cs | 45 ++++++++----------- 1 file changed, 18 insertions(+), 27 deletions(-) diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/N64/NativeApi/mupen64plusCoreApi.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/N64/NativeApi/mupen64plusCoreApi.cs index 12a120e750..e735100b0f 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/N64/NativeApi/mupen64plusCoreApi.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/N64/NativeApi/mupen64plusCoreApi.cs @@ -1,5 +1,6 @@ using System; using System.Collections.Generic; +using System.Reflection; using System.Runtime.InteropServices; using System.Threading; @@ -27,8 +28,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.N64.NativeApi bool event_frameend = false; bool event_breakpoint = false; - private static readonly OSTailoredCode.ILinkedLibManager libLoader = OSTailoredCode.LinkedLibManager; - public enum m64p_error { M64ERR_SUCCESS = 0, @@ -471,8 +470,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.N64.NativeApi delegate m64p_error DebugStep(); DebugStep m64pDebugStep; - // DLL handles - public IntPtr CoreDll { get; } + private readonly DynamicLibraryImportResolver Library = new DynamicLibraryImportResolver(OSTailoredCode.IsUnixHost ? "libmupen64plus.so.2" : "mupen64plus.dll"); public mupen64plusApi(N64 bizhawkCore, byte[] rom, VideoPluginSettings video_settings, int SaveType, int CoreType, bool DisableExpansionSlot) { @@ -484,8 +482,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.N64.NativeApi } this.bizhawkCore = bizhawkCore; - CoreDll = libLoader.LoadOrThrow("mupen64plus"); - connectFunctionPointers(); // Start up the core @@ -576,14 +572,14 @@ namespace BizHawk.Emulation.Cores.Nintendo.N64.NativeApi cb.GetType(); } - internal static T GetTypedDelegate(IntPtr lib, string proc) where T : Delegate => (T) Marshal.GetDelegateForFunctionPointer(libLoader.GetProcAddrOrThrow(lib, proc), typeof(T)); + internal static T GetTypedDelegate(IntPtr lib, string proc) where T : Delegate => (T) Marshal.GetDelegateForFunctionPointer(OSTailoredCode.LinkedLibManager.GetProcAddrOrThrow(lib, proc), typeof(T)); /// /// Look up function pointers in the dlls /// void connectFunctionPointers() { - T GetCoreDelegate(string proc) where T : Delegate => GetTypedDelegate(CoreDll, proc); + T GetCoreDelegate(string proc) where T : Delegate => (T) Marshal.GetDelegateForFunctionPointer(Library.GetProcAddrOrThrow(proc), typeof(T)); m64pCoreStartup = GetCoreDelegate("CoreStartup"); m64pCoreShutdown = GetCoreDelegate("CoreShutdown"); @@ -969,40 +965,35 @@ namespace BizHawk.Emulation.Cores.Nintendo.N64.NativeApi m64pCoreDoCommandPtr(m64p_command.M64CMD_ROM_CLOSE, 0, IntPtr.Zero); m64pCoreShutdown(); - libLoader.FreeByPtr(CoreDll); + Library.Dispose(); disposed = true; } } + private static readonly FieldInfo fiDLIRInternalPtr = typeof(DynamicLibraryImportResolver).GetField("_p", BindingFlags.Instance | BindingFlags.NonPublic); + struct AttachedPlugin { - public PluginStartup dllStartup; + public DynamicLibraryImportResolver dllThinWrapper; public PluginShutdown dllShutdown; - public IntPtr dllHandle; } Dictionary plugins = new Dictionary(); public IntPtr AttachPlugin(m64p_plugin_type type, string PluginName) { - if (plugins.ContainsKey(type)) - DetachPlugin(type); - - AttachedPlugin plugin; - plugin.dllHandle = libLoader.LoadOrThrow(PluginName); - plugin.dllStartup = GetTypedDelegate(plugin.dllHandle, "PluginStartup"); - plugin.dllShutdown = GetTypedDelegate(plugin.dllHandle, "PluginShutdown"); - plugin.dllStartup(CoreDll, null, null); - - m64p_error result = m64pCoreAttachPlugin(type, plugin.dllHandle); - if (result != m64p_error.M64ERR_SUCCESS) + static IntPtr GetDLIRPtrByRefl(DynamicLibraryImportResolver dlir) => (IntPtr) fiDLIRInternalPtr.GetValue(dlir); + if (plugins.ContainsKey(type)) DetachPlugin(type); + var lib = new DynamicLibraryImportResolver(PluginName); + var libPtr = GetDLIRPtrByRefl(lib); + GetTypedDelegate(libPtr, "PluginStartup")(GetDLIRPtrByRefl(Library), null, null); + if (m64pCoreAttachPlugin(type, libPtr) != m64p_error.M64ERR_SUCCESS) { - libLoader.FreeByPtr(plugin.dllHandle); + lib.Dispose(); throw new InvalidOperationException($"Error during attaching plugin {PluginName}"); } - - plugins.Add(type, plugin); - return plugin.dllHandle; + plugins.Add(type, new AttachedPlugin { dllThinWrapper = lib, dllShutdown = GetTypedDelegate(libPtr, "PluginShutdown") }); + return libPtr; } public void DetachPlugin(m64p_plugin_type type) @@ -1012,7 +1003,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.N64.NativeApi plugins.Remove(type); m64pCoreDetachPlugin(type); plugin.dllShutdown(); - libLoader.FreeByPtr(plugin.dllHandle); + plugin.dllThinWrapper.Dispose(); } }