From e19f39e08c20d333660cc54242a631fff6974381 Mon Sep 17 00:00:00 2001 From: zeromus Date: Mon, 29 Oct 2012 08:37:22 +0000 Subject: [PATCH] support moving all dlls to the dll directory by adding assembly binding resolve redirection --- BizHawk.MultiClient/Program.cs | 58 ++++++++++++---------- BizHawk.MultiClient/output/MakeRelease.bat | 11 +++- BizHawk.Util/Win32.cs | 3 -- DiscoHawk/DiscoHawk.cs | 47 +++++++++++++++--- DiscoHawk/DiscoHawk.csproj | 5 +- 5 files changed, 83 insertions(+), 41 deletions(-) diff --git a/BizHawk.MultiClient/Program.cs b/BizHawk.MultiClient/Program.cs index 756b1f94b8..de9b64aee8 100644 --- a/BizHawk.MultiClient/Program.cs +++ b/BizHawk.MultiClient/Program.cs @@ -1,4 +1,7 @@ using System; +using System.IO; +using System.Reflection; +using System.Runtime.InteropServices; using System.Windows.Forms; #if WINDOWS using SlimDX.Direct3D9; @@ -13,33 +16,19 @@ namespace BizHawk.MultiClient [STAThread] unsafe static void Main(string[] args) { - //string test = BizHawk.Emulation.Consoles.Nintendo.SNES.LibsnesDll.snes_library_revision_minor().ToString(); +#if WINDOWS + // this will look in subdirectory "dll" to load pinvoked stuff + SetDllDirectory(System.IO.Path.Combine(PathManager.GetExeDirectoryAbsolute(), "dll")); - //BizHawk.Emulation.Consoles.Nintendo.SNES.LibsnesDll.snes_init(); + //in case assembly resolution fails, such as if we moved them into the dll subdiretory, this event handler can reroute to them + AppDomain.CurrentDomain.AssemblyResolve += new ResolveEventHandler(CurrentDomain_AssemblyResolve); +#endif - //BizHawk.Emulation.Consoles.Nintendo.SNES.LibsnesDll.snes_video_refresh_t myvidproc = - // (ushort* data, int width, int height) => - // { - // }; - - //System.Runtime.InteropServices.GCHandle handle = System.Runtime.InteropServices.GCHandle.Alloc(myvidproc); - ////IntPtr ip = System.Runtime.InteropServices.Marshal.GetFunctionPointerForDelegate(myvidproc); - - //BizHawk.Emulation.Consoles.Nintendo.SNES.LibsnesDll.snes_set_video_refresh(myvidproc); - - - //byte[] rom = System.IO.File.ReadAllBytes("d:\\Super Mario World (US).smc"); - //BizHawk.Emulation.Consoles.Nintendo.SNES.LibsnesDll.snes_load_cartridge_normal(null, rom, rom.Length); - - //BizHawk.Emulation.Consoles.Nintendo.SNES.LibsnesDll.snes_power(); - - //int framectr = 0; - //for (; ; ) - //{ - // framectr++; - // BizHawk.Emulation.Consoles.Nintendo.SNES.LibsnesDll.snes_run(); - //} + SubMain(args); + } + static void SubMain(string[] args) + { Application.EnableVisualStyles(); Application.SetCompatibleTextRenderingDefault(false); @@ -47,9 +36,6 @@ namespace BizHawk.MultiClient Global.Config = ConfigService.Load(PathManager.DefaultIniPath, new Config()); #if WINDOWS - // this will look in subdirectory "dll" to load pinvoked stuff - Win32.SetDllDirectory(System.IO.Path.Combine(PathManager.GetExeDirectoryAbsolute(),"dll")); - try { Global.DSound = new DirectSound(); } catch { @@ -102,6 +88,24 @@ namespace BizHawk.MultiClient #endif } + //declared here instead of a more usual place to avoid dependencies on the more usual place +#if WINDOWS + [DllImport("kernel32.dll", SetLastError = true)] + static extern bool SetDllDirectory(string lpPathName); +#endif + + + static Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args) + { + //load missing assemblies by trying to find them in the dll directory + string dllname = new AssemblyName(args.Name).Name + ".dll"; + string directory = System.IO.Path.Combine(PathManager.GetExeDirectoryAbsolute(), "dll"); + string fname = Path.Combine(directory, dllname); + if (!File.Exists(fname)) return null; + //it is important that we use LoadFile here and not load from a byte array; otherwise mixed (managed/unamanged) assemblies can't load + return Assembly.LoadFile(fname); + } + #if WINDOWS public class SingleInstanceController : WindowsFormsApplicationBase { diff --git a/BizHawk.MultiClient/output/MakeRelease.bat b/BizHawk.MultiClient/output/MakeRelease.bat index 0320922e29..fbf358635f 100644 --- a/BizHawk.MultiClient/output/MakeRelease.bat +++ b/BizHawk.MultiClient/output/MakeRelease.bat @@ -1,3 +1,12 @@ del /s BizHawk.zip + +rem slimdx has a way of not making it into the output directory, so this is a good time to make sure copy ..\..\SlimDx.dll -zip -X -9 -r BizHawk.zip BizHawk.MultiClient.exe DiscoHawk.exe *.dll dll ffmpeg.exe gamedb NES\Palettes Lua Gameboy\Palettes \ No newline at end of file + +rem at the present we support moving all these dlls into the dll subdirectory +rem that might be troublesome some day..... if it does get troublesome, then we'll have to +rem explicitly list the OK ones here as individual copies. until then.... + +copy *.dll dll + +zip -X -9 -r BizHawk.zip BizHawk.MultiClient.exe DiscoHawk.exe dll gamedb NES\Palettes Lua Gameboy\Palettes diff --git a/BizHawk.Util/Win32.cs b/BizHawk.Util/Win32.cs index aebcf806cb..3d8027554d 100644 --- a/BizHawk.Util/Win32.cs +++ b/BizHawk.Util/Win32.cs @@ -439,9 +439,6 @@ namespace BizHawk [DllImport("user32.dll", CharSet = CharSet.Auto)] public static extern int SendMessage(IntPtr hWnd, int msg, int wParam, int lParam); - - [DllImport("kernel32.dll", SetLastError = true)] - public static extern bool SetDllDirectory(string lpPathName); } } \ No newline at end of file diff --git a/DiscoHawk/DiscoHawk.cs b/DiscoHawk/DiscoHawk.cs index 108533cc38..1931b5adad 100644 --- a/DiscoHawk/DiscoHawk.cs +++ b/DiscoHawk/DiscoHawk.cs @@ -1,4 +1,5 @@ using System; +using System.Runtime.InteropServices; using System.Reflection; using System.Collections.Generic; using System.IO; @@ -17,19 +18,49 @@ namespace BizHawk public static string GetExeDirectoryAbsolute() { - string p = Path.GetDirectoryName(Assembly.GetEntryAssembly().GetName().CodeBase); - if (p.Substring(0, 6) == "file:\\") - p = p.Remove(0, 6); - string z = p; - return p; + var uri = new Uri(Assembly.GetEntryAssembly().GetName().CodeBase); + string module = uri.LocalPath + System.Web.HttpUtility.UrlDecode(uri.Fragment); + return Path.GetDirectoryName(module); } + static Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args) + { + //load missing assemblies by trying to find them in the dll directory + string dllname = new AssemblyName(args.Name).Name + ".dll"; + string directory = System.IO.Path.Combine(GetExeDirectoryAbsolute(), "dll"); + string fname = Path.Combine(directory, dllname); + if (!File.Exists(fname)) return null; + //it is important that we use LoadFile here and not load from a byte array; otherwise mixed (managed/unamanged) assemblies can't load + return Assembly.LoadFile(fname); + } + + //declared here instead of a more usual place to avoid dependencies on the more usual place +#if WINDOWS + [DllImport("kernel32.dll", SetLastError = true)] + static extern bool SetDllDirectory(string lpPathName); +#endif + [STAThread] static void Main(string[] args) { - var ffmpegPath = Path.Combine(GetExeDirectoryAbsolute(), "ffmpeg.exe"); - DiscSystem.FFMpeg.FFMpegPath = ffmpegPath; - AudioExtractor.FFmpegPath = ffmpegPath; +#if WINDOWS + // this will look in subdirectory "dll" to load pinvoked stuff + SetDllDirectory(System.IO.Path.Combine(GetExeDirectoryAbsolute(), "dll")); + + //in case assembly resolution fails, such as if we moved them into the dll subdiretory, this event handler can reroute to them + AppDomain.CurrentDomain.AssemblyResolve += new ResolveEventHandler(CurrentDomain_AssemblyResolve); +#endif + + SubMain(args); + } + + static void SubMain(string[] args) + { + var ffmpegPath = Path.Combine(GetExeDirectoryAbsolute(), "ffmpeg.exe"); + if(!File.Exists(ffmpegPath)) + ffmpegPath = Path.Combine(Path.Combine(GetExeDirectoryAbsolute(), "dll"), "ffmpeg.exe"); + DiscSystem.FFMpeg.FFMpegPath = ffmpegPath; + AudioExtractor.FFmpegPath = ffmpegPath; new DiscoHawk().Run(args); } diff --git a/DiscoHawk/DiscoHawk.csproj b/DiscoHawk/DiscoHawk.csproj index 83b1256009..602fff2011 100644 --- a/DiscoHawk/DiscoHawk.csproj +++ b/DiscoHawk/DiscoHawk.csproj @@ -38,7 +38,7 @@ full false ..\BizHawk.MultiClient\output\ - DEBUG;TRACE + TRACE;DEBUG;WINDOWS prompt 4 false @@ -49,7 +49,7 @@ pdbonly true ..\BizHawk.MultiClient\output\ - TRACE + TRACE;WINDOWS prompt 4 false @@ -110,6 +110,7 @@ +