From 75e815f09e771bc4e6e2b98c41f67c1d1d3e38c6 Mon Sep 17 00:00:00 2001 From: YoshiRulz Date: Tue, 28 Jul 2020 17:36:34 +1000 Subject: [PATCH] A little cleanup of DLIR --- src/BizHawk.Common/IImportResolver.cs | 41 ++++++++++--------- .../Sound/SpeexResampler.cs | 2 +- .../Consoles/Nintendo/GBA/MGBAHawk.cs | 2 +- .../Consoles/Nintendo/QuickNES/QuickNES.cs | 2 +- .../Waterbox/WaterboxHost.cs | 2 +- 5 files changed, 26 insertions(+), 23 deletions(-) diff --git a/src/BizHawk.Common/IImportResolver.cs b/src/BizHawk.Common/IImportResolver.cs index e9fde5f51a..62b361356c 100644 --- a/src/BizHawk.Common/IImportResolver.cs +++ b/src/BizHawk.Common/IImportResolver.cs @@ -3,6 +3,7 @@ using System.Collections.Generic; using System.IO; using System.Linq; using System.Reflection; +using System.Runtime.InteropServices; namespace BizHawk.Common { @@ -18,29 +19,32 @@ namespace BizHawk.Common public class DynamicLibraryImportResolver : IDisposable, IImportResolver { - private static readonly Lazy> asdf = new Lazy>(() => + private static readonly IReadOnlyCollection UnixSearchPaths; + + static DynamicLibraryImportResolver() { var currDir = Path.GetDirectoryName(Assembly.GetExecutingAssembly().CodeBase)?.Replace("file:", "") ?? string.Empty; - return new[] { "/usr/lib/", "/usr/lib/bizhawk/", "./", "./dll/" }.Select(dir => dir[0] == '.' ? currDir + dir.Substring(1) : dir); - }); + UnixSearchPaths = new[] + { + "/usr/lib/", "/usr/lib/bizhawk/", + $"{currDir}/", $"{currDir}/dll/" + }; + } + + private static string UnixResolveFilePath(string orig) => orig[0] == '/' + ? orig + : UnixSearchPaths.Select(dir => dir + orig).FirstOrDefault(File.Exists) ?? orig; private IntPtr _p; - private bool _eternal; - /// - /// - /// - /// - /// - /// If true, the DLL will never be unloaded, even by god. Use this when you need lifetime semantics similar to [DllImport] - /// - public DynamicLibraryImportResolver(string dllName, bool eternal = false) + public readonly bool HasLimitedLifetime; + + /// will never be unloaded iff false (like [DllImport]) + public DynamicLibraryImportResolver(string dllName, bool hasLimitedLifetime = true) { - static string ResolveFilePath(string orig) => orig[0] == '/' ? orig : asdf.Value.Select(dir => dir + orig).FirstOrDefault(File.Exists) ?? orig; - _p = OSTailoredCode.LinkedLibManager.LoadOrThrow(OSTailoredCode.IsUnixHost ? ResolveFilePath(dllName) : dllName); - _eternal = eternal; - if (eternal) - GC.SuppressFinalize(this); + _p = OSTailoredCode.LinkedLibManager.LoadOrThrow(OSTailoredCode.IsUnixHost ? UnixResolveFilePath(dllName) : dllName); // on Windows, EmuHawk modifies its process' search path + HasLimitedLifetime = hasLimitedLifetime; + if (!hasLimitedLifetime) GC.SuppressFinalize(this); } public IntPtr GetProcAddrOrZero(string entryPoint) => OSTailoredCode.LinkedLibManager.GetProcAddrOrZero(_p, entryPoint); @@ -49,8 +53,7 @@ namespace BizHawk.Common private void DisposeHelper() { - if (_eternal || _p == IntPtr.Zero) - return; + if (!HasLimitedLifetime || _p == IntPtr.Zero) return; OSTailoredCode.LinkedLibManager.FreeByPtr(_p); _p = IntPtr.Zero; } diff --git a/src/BizHawk.Emulation.Common/Sound/SpeexResampler.cs b/src/BizHawk.Emulation.Common/Sound/SpeexResampler.cs index f047ee28ce..5fb8115bf1 100644 --- a/src/BizHawk.Emulation.Common/Sound/SpeexResampler.cs +++ b/src/BizHawk.Emulation.Common/Sound/SpeexResampler.cs @@ -18,7 +18,7 @@ namespace BizHawk.Emulation.Common static SpeexResampler() { var resolver = new DynamicLibraryImportResolver( - OSTailoredCode.IsUnixHost ? "libspeexdsp.so.1" : "libspeexdsp.dll", eternal: true); + OSTailoredCode.IsUnixHost ? "libspeexdsp.so.1" : "libspeexdsp.dll", hasLimitedLifetime: false); NativeDSP = BizInvoker.GetInvoker(resolver, CallingConventionAdapters.Native); } diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/MGBAHawk.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/MGBAHawk.cs index a655ce0cca..d7dc7d2783 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/MGBAHawk.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/MGBAHawk.cs @@ -18,7 +18,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBA static MGBAHawk() { var resolver = new DynamicLibraryImportResolver( - OSTailoredCode.IsUnixHost ? "libmgba.dll.so" : "mgba.dll", eternal: true); + OSTailoredCode.IsUnixHost ? "libmgba.dll.so" : "mgba.dll", hasLimitedLifetime: false); LibmGBA = BizInvoker.GetInvoker(resolver, CallingConventionAdapters.Native); } diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/QuickNES/QuickNES.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/QuickNES/QuickNES.cs index a0d862fcee..6c0d43b6c2 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/QuickNES/QuickNES.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/QuickNES/QuickNES.cs @@ -28,7 +28,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Nintendo.QuickNES static QuickNES() { var resolver = new DynamicLibraryImportResolver( - $"libquicknes{(OSTailoredCode.IsUnixHost ? ".dll.so.0.7.0" : ".dll")}", eternal: true); + $"libquicknes{(OSTailoredCode.IsUnixHost ? ".dll.so.0.7.0" : ".dll")}", hasLimitedLifetime: false); QN = BizInvoker.GetInvoker(resolver, CallingConventionAdapters.Native); QN.qn_setup_mappers(); } diff --git a/src/BizHawk.Emulation.Cores/Waterbox/WaterboxHost.cs b/src/BizHawk.Emulation.Cores/Waterbox/WaterboxHost.cs index 79dd00afee..4317dae7ce 100644 --- a/src/BizHawk.Emulation.Cores/Waterbox/WaterboxHost.cs +++ b/src/BizHawk.Emulation.Cores/Waterbox/WaterboxHost.cs @@ -78,7 +78,7 @@ namespace BizHawk.Emulation.Cores.Waterbox static WaterboxHost() { NativeImpl = BizInvoker.GetInvoker( - new DynamicLibraryImportResolver(OSTailoredCode.IsUnixHost ? "libwaterboxhost.so" : "waterboxhost.dll", eternal: true), + new DynamicLibraryImportResolver(OSTailoredCode.IsUnixHost ? "libwaterboxhost.so" : "waterboxhost.dll", hasLimitedLifetime: false), CallingConventionAdapters.Native); #if !DEBUG NativeImpl.wbx_set_always_evict_blocks(false);