Misc. refactoring
This commit is contained in:
parent
752f8cfd49
commit
436b6452fb
|
@ -1,5 +1,4 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
@ -15,7 +14,7 @@ using BizHawk.Client.Common;
|
||||||
|
|
||||||
namespace BizHawk.Client.EmuHawk
|
namespace BizHawk.Client.EmuHawk
|
||||||
{
|
{
|
||||||
static class Program
|
internal static class Program
|
||||||
{
|
{
|
||||||
static Program()
|
static Program()
|
||||||
{
|
{
|
||||||
|
@ -37,27 +36,24 @@ namespace BizHawk.Client.EmuHawk
|
||||||
var vc2012 = libLoader.LoadPlatformSpecific("msvcr120.dll"); //TODO - check version?
|
var vc2012 = libLoader.LoadPlatformSpecific("msvcr120.dll"); //TODO - check version?
|
||||||
var vc2010 = libLoader.LoadPlatformSpecific("msvcr100.dll"); //TODO - check version?
|
var vc2010 = libLoader.LoadPlatformSpecific("msvcr100.dll"); //TODO - check version?
|
||||||
var vc2010p = libLoader.LoadPlatformSpecific("msvcp100.dll");
|
var vc2010p = libLoader.LoadPlatformSpecific("msvcp100.dll");
|
||||||
bool fail = false, warn = false;
|
var fail = vc2015 == IntPtr.Zero || vc2010 == IntPtr.Zero || vc2012 == IntPtr.Zero || vc2010p == IntPtr.Zero;
|
||||||
warn |= d3dx9 == IntPtr.Zero;
|
var warn = d3dx9 == IntPtr.Zero;
|
||||||
fail |= vc2015 == IntPtr.Zero;
|
|
||||||
fail |= vc2010 == IntPtr.Zero;
|
|
||||||
fail |= vc2012 == IntPtr.Zero;
|
|
||||||
fail |= vc2010p == IntPtr.Zero;
|
|
||||||
if (fail || warn)
|
if (fail || warn)
|
||||||
{
|
{
|
||||||
var sw = new System.IO.StringWriter();
|
var alertLines = new[]
|
||||||
sw.WriteLine("[ OK ] .Net 4.6.1 (You couldn't even get here without it)");
|
{
|
||||||
sw.WriteLine("[{0}] Direct3d 9", d3dx9 == IntPtr.Zero ? "FAIL" : " OK ");
|
"[ OK ] .NET CLR (You wouldn't even get here without it)",
|
||||||
sw.WriteLine("[{0}] Visual C++ 2010 SP1 Runtime", (vc2010 == IntPtr.Zero || vc2010p == IntPtr.Zero) ? "FAIL" : " OK ");
|
$"[{(d3dx9 == IntPtr.Zero ? "WARN" : " OK ")}] Direct3d 9",
|
||||||
sw.WriteLine("[{0}] Visual C++ 2012 Runtime", (vc2012 == IntPtr.Zero) ? "FAIL" : " OK ");
|
$"[{(vc2010 == IntPtr.Zero || vc2010p == IntPtr.Zero ? "FAIL" : " OK ")}] Visual C++ 2010 SP1 Runtime",
|
||||||
sw.WriteLine("[{0}] Visual C++ 2015 Runtime", (vc2015 == IntPtr.Zero) ? "FAIL" : " OK ");
|
$"[{(vc2012 == IntPtr.Zero ? "FAIL" : " OK ")}] Visual C++ 2012 Runtime",
|
||||||
var str = sw.ToString();
|
$"[{(vc2015 == IntPtr.Zero ? "FAIL" : " OK ")}] Visual C++ 2015 Runtime"
|
||||||
var box = new BizHawk.Client.EmuHawk.CustomControls.PrereqsAlert(!fail);
|
};
|
||||||
box.textBox1.Text = str;
|
var box = new BizHawk.Client.EmuHawk.CustomControls.PrereqsAlert(!fail)
|
||||||
|
{
|
||||||
|
textBox1 = { Text = string.Concat("\n", alertLines) }
|
||||||
|
};
|
||||||
box.ShowDialog();
|
box.ShowDialog();
|
||||||
if (!fail) { }
|
if (fail) System.Diagnostics.Process.GetCurrentProcess().Kill();
|
||||||
else
|
|
||||||
System.Diagnostics.Process.GetCurrentProcess().Kill();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
libLoader.FreePlatformSpecific(d3dx9);
|
libLoader.FreePlatformSpecific(d3dx9);
|
||||||
|
@ -67,11 +63,11 @@ namespace BizHawk.Client.EmuHawk
|
||||||
libLoader.FreePlatformSpecific(vc2010p);
|
libLoader.FreePlatformSpecific(vc2010p);
|
||||||
|
|
||||||
// this will look in subdirectory "dll" to load pinvoked stuff
|
// this will look in subdirectory "dll" to load pinvoked stuff
|
||||||
string dllDir = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "dll");
|
var dllDir = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "dll");
|
||||||
SetDllDirectory(dllDir);
|
SetDllDirectory(dllDir);
|
||||||
|
|
||||||
//in case assembly resolution fails, such as if we moved them into the dll subdiretory, this event handler can reroute to them
|
//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);
|
AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;
|
||||||
|
|
||||||
//but before we even try doing that, whack the MOTW from everything in that directory (thats a dll)
|
//but before we even try doing that, whack the MOTW from everything in that directory (thats a dll)
|
||||||
//otherwise, some people will have crashes at boot-up due to .net security disliking MOTW.
|
//otherwise, some people will have crashes at boot-up due to .net security disliking MOTW.
|
||||||
|
@ -83,14 +79,14 @@ namespace BizHawk.Client.EmuHawk
|
||||||
}
|
}
|
||||||
|
|
||||||
[STAThread]
|
[STAThread]
|
||||||
static int Main(string[] args)
|
private static int Main(string[] args)
|
||||||
{
|
{
|
||||||
return SubMain(args);
|
return SubMain(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
//NoInlining should keep this code from getting jammed into Main() which would create dependencies on types which havent been setup by the resolver yet... or something like that
|
//NoInlining should keep this code from getting jammed into Main() which would create dependencies on types which havent been setup by the resolver yet... or something like that
|
||||||
[System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.NoInlining)]
|
[System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.NoInlining)]
|
||||||
static int SubMain(string[] args)
|
private static int SubMain(string[] args)
|
||||||
{
|
{
|
||||||
// this check has to be done VERY early. i stepped through a debug build with wrong .dll versions purposely used,
|
// this check has to be done VERY early. i stepped through a debug build with wrong .dll versions purposely used,
|
||||||
// and there was a TypeLoadException before the first line of SubMain was reached (some static ColorType init?)
|
// and there was a TypeLoadException before the first line of SubMain was reached (some static ColorType init?)
|
||||||
|
@ -112,7 +108,7 @@ namespace BizHawk.Client.EmuHawk
|
||||||
|
|
||||||
HawkFile.ArchiveHandlerFactory = new SevenZipSharpArchiveHandler();
|
HawkFile.ArchiveHandlerFactory = new SevenZipSharpArchiveHandler();
|
||||||
|
|
||||||
ArgParser argParser = new ArgParser();
|
var argParser = new ArgParser();
|
||||||
argParser.ParseArguments(args);
|
argParser.ParseArguments(args);
|
||||||
if (argParser.cmdConfigFile != null) PathManager.SetDefaultIniPath(argParser.cmdConfigFile);
|
if (argParser.cmdConfigFile != null) PathManager.SetDefaultIniPath(argParser.cmdConfigFile);
|
||||||
|
|
||||||
|
@ -132,13 +128,9 @@ namespace BizHawk.Client.EmuHawk
|
||||||
BizHawk.Client.Common.StringLogUtil.DefaultToAWE = Global.Config.MoviesInAWE;
|
BizHawk.Client.Common.StringLogUtil.DefaultToAWE = Global.Config.MoviesInAWE;
|
||||||
|
|
||||||
// super hacky! this needs to be done first. still not worth the trouble to make this system fully proper
|
// super hacky! this needs to be done first. still not worth the trouble to make this system fully proper
|
||||||
for (int i = 0; i < args.Length; i++)
|
if (Array.Exists(args, arg => arg.StartsWith("--gdi", StringComparison.InvariantCultureIgnoreCase)))
|
||||||
{
|
{
|
||||||
var arg = args[i].ToLower();
|
Global.Config.DispMethod = Config.EDispMethod.GdiPlus;
|
||||||
if (arg.StartsWith("--gdi"))
|
|
||||||
{
|
|
||||||
Global.Config.DispMethod = Config.EDispMethod.GdiPlus;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// create IGL context. we do this whether or not the user has selected OpenGL, so that we can run opengl-based emulator cores
|
// create IGL context. we do this whether or not the user has selected OpenGL, so that we can run opengl-based emulator cores
|
||||||
|
@ -161,8 +153,7 @@ namespace BizHawk.Client.EmuHawk
|
||||||
}
|
}
|
||||||
catch(Exception ex)
|
catch(Exception ex)
|
||||||
{
|
{
|
||||||
var e2 = new Exception("Initialization of Direct3d 9 Display Method failed; falling back to GDI+", ex);
|
new ExceptionBox(new Exception("Initialization of Direct3d 9 Display Method failed; falling back to GDI+", ex)).ShowDialog();
|
||||||
new ExceptionBox(e2).ShowDialog();
|
|
||||||
|
|
||||||
// fallback
|
// fallback
|
||||||
Global.Config.DispMethod = Config.EDispMethod.GdiPlus;
|
Global.Config.DispMethod = Config.EDispMethod.GdiPlus;
|
||||||
|
@ -174,8 +165,7 @@ namespace BizHawk.Client.EmuHawk
|
||||||
GlobalWin.GL = GlobalWin.IGL_GL;
|
GlobalWin.GL = GlobalWin.IGL_GL;
|
||||||
|
|
||||||
// check the opengl version and dont even try to boot this crap up if its too old
|
// check the opengl version and dont even try to boot this crap up if its too old
|
||||||
int version = GlobalWin.IGL_GL.Version;
|
if (GlobalWin.IGL_GL.Version < 200)
|
||||||
if (version < 200)
|
|
||||||
{
|
{
|
||||||
// fallback
|
// fallback
|
||||||
Global.Config.DispMethod = Config.EDispMethod.GdiPlus;
|
Global.Config.DispMethod = Config.EDispMethod.GdiPlus;
|
||||||
|
@ -190,8 +180,8 @@ namespace BizHawk.Client.EmuHawk
|
||||||
}
|
}
|
||||||
catch(Exception ex)
|
catch(Exception ex)
|
||||||
{
|
{
|
||||||
var e2 = new Exception("Initialization of Display Method failed; falling back to GDI+", ex);
|
new ExceptionBox(new Exception("Initialization of Display Method failed; falling back to GDI+", ex)).ShowDialog();
|
||||||
new ExceptionBox(e2).ShowDialog();
|
|
||||||
//fallback
|
//fallback
|
||||||
Global.Config.DispMethod = Config.EDispMethod.GdiPlus;
|
Global.Config.DispMethod = Config.EDispMethod.GdiPlus;
|
||||||
goto REDO_DISPMETHOD;
|
goto REDO_DISPMETHOD;
|
||||||
|
@ -204,7 +194,7 @@ namespace BizHawk.Client.EmuHawk
|
||||||
//It isn't clear whether we need the earlier SetDllDirectory(), but I think we do.
|
//It isn't clear whether we need the earlier SetDllDirectory(), but I think we do.
|
||||||
//note: this is pasted instead of being put in a static method due to this initialization code being sensitive to things like that, and not wanting to cause it to break
|
//note: this is pasted instead of being put in a static method due to this initialization code being sensitive to things like that, and not wanting to cause it to break
|
||||||
//pasting should be safe (not affecting the jit order of things)
|
//pasting should be safe (not affecting the jit order of things)
|
||||||
string dllDir = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "dll");
|
var dllDir = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "dll");
|
||||||
SetDllDirectory(dllDir);
|
SetDllDirectory(dllDir);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -214,7 +204,7 @@ namespace BizHawk.Client.EmuHawk
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
new SingleInstanceController(args).Run(args);
|
new SingleInstanceController(args).Run();
|
||||||
}
|
}
|
||||||
catch (ObjectDisposedException)
|
catch (ObjectDisposedException)
|
||||||
{
|
{
|
||||||
|
@ -273,20 +263,21 @@ namespace BizHawk.Client.EmuHawk
|
||||||
return GlobalWin.ExitCode;
|
return GlobalWin.ExitCode;
|
||||||
} //SubMain
|
} //SubMain
|
||||||
|
|
||||||
//declared here instead of a more usual place to avoid dependencies on the more usual place
|
|
||||||
#if WINDOWS
|
#if WINDOWS
|
||||||
|
//declared here instead of a more usual place to avoid dependencies on the more usual place
|
||||||
|
|
||||||
[DllImport("kernel32.dll", SetLastError = true)]
|
[DllImport("kernel32.dll", SetLastError = true)]
|
||||||
static extern uint SetDllDirectory(string lpPathName);
|
private static extern uint SetDllDirectory(string lpPathName);
|
||||||
|
|
||||||
[DllImport("kernel32.dll", EntryPoint = "DeleteFileW", SetLastError = true, CharSet = CharSet.Unicode, ExactSpelling = true)]
|
[DllImport("kernel32.dll", EntryPoint = "DeleteFileW", SetLastError = true, CharSet = CharSet.Unicode, ExactSpelling = true)]
|
||||||
static extern bool DeleteFileW([MarshalAs(UnmanagedType.LPWStr)]string lpFileName);
|
private static extern bool DeleteFileW([MarshalAs(UnmanagedType.LPWStr)]string lpFileName);
|
||||||
|
|
||||||
public static void RemoveMOTW(string path)
|
private static void RemoveMOTW(string path)
|
||||||
{
|
{
|
||||||
DeleteFileW($"{path}:Zone.Identifier");
|
DeleteFileW($"{path}:Zone.Identifier");
|
||||||
}
|
}
|
||||||
|
|
||||||
static void WhackAllMOTW(string dllDir)
|
private static void WhackAllMOTW(string dllDir)
|
||||||
{
|
{
|
||||||
var todo = new Queue<DirectoryInfo>(new[] { new DirectoryInfo(dllDir) });
|
var todo = new Queue<DirectoryInfo>(new[] { new DirectoryInfo(dllDir) });
|
||||||
while (todo.Count > 0)
|
while (todo.Count > 0)
|
||||||
|
@ -298,14 +289,12 @@ namespace BizHawk.Client.EmuHawk
|
||||||
foreach (var fi in di.GetFiles("*.exe"))
|
foreach (var fi in di.GetFiles("*.exe"))
|
||||||
RemoveMOTW(fi.FullName);
|
RemoveMOTW(fi.FullName);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
private static Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
|
||||||
static Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
|
|
||||||
{
|
{
|
||||||
string requested = args.Name;
|
var requested = args.Name;
|
||||||
|
|
||||||
//mutate filename depending on selection of lua core. here's how it works
|
//mutate filename depending on selection of lua core. here's how it works
|
||||||
//1. we build NLua to the output/dll/lua directory. that brings KopiLua with it
|
//1. we build NLua to the output/dll/lua directory. that brings KopiLua with it
|
||||||
|
@ -313,52 +302,43 @@ namespace BizHawk.Client.EmuHawk
|
||||||
//3. When NLua assembly attempts to load, it can't find it
|
//3. When NLua assembly attempts to load, it can't find it
|
||||||
//I. if LuaInterface is selected by the user, we switch to requesting that.
|
//I. if LuaInterface is selected by the user, we switch to requesting that.
|
||||||
// (those DLLs are built into the output/DLL directory)
|
// (those DLLs are built into the output/DLL directory)
|
||||||
//II. if NLua is selected by the user, we skip over this part;
|
//II. if NLua is selected by the user, we skip over this part;
|
||||||
// later, we look for NLua or KopiLua assembly names and redirect them to files located in the output/DLL/nlua directory
|
// later, we look for NLua or KopiLua assembly names and redirect them to files located in the output/DLL/nlua directory
|
||||||
if (new AssemblyName(requested).Name == "NLua")
|
if (new AssemblyName(requested).Name == "NLua")
|
||||||
{
|
{
|
||||||
//this method referencing Global.Config makes assemblies get loaded, which isnt smart from the assembly resolver.
|
//this method referencing Global.Config makes assemblies get loaded, which isnt smart from the assembly resolver.
|
||||||
//so.. we're going to resort to something really bad.
|
//so.. we're going to resort to something really bad.
|
||||||
//avert your eyes.
|
//avert your eyes.
|
||||||
bool UseNLua = true;
|
var configPath = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "config.ini");
|
||||||
string configPath = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "config.ini");
|
if (File.Exists(configPath)
|
||||||
if (File.Exists(configPath))
|
&& (Array.Find(File.ReadAllLines(configPath), line => line.Contains(" \"UseNLua\": ")) ?? string.Empty)
|
||||||
|
.Contains("false"))
|
||||||
{
|
{
|
||||||
var cfg = File.ReadAllLines(configPath);
|
requested = "LuaInterface";
|
||||||
var usenlua_key = cfg.FirstOrDefault(line=>line.Contains(" \"UseNLua\": "));
|
|
||||||
if (usenlua_key != null)
|
|
||||||
if (usenlua_key.Contains("false"))
|
|
||||||
UseNLua = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (UseNLua) { }
|
|
||||||
else requested = "LuaInterface";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
lock (AppDomain.CurrentDomain)
|
lock (AppDomain.CurrentDomain)
|
||||||
{
|
{
|
||||||
var asms = AppDomain.CurrentDomain.GetAssemblies();
|
var firstAsm = Array.Find(AppDomain.CurrentDomain.GetAssemblies(), asm => asm.FullName == requested);
|
||||||
foreach (var asm in asms)
|
if (firstAsm != null) return firstAsm;
|
||||||
if (asm.FullName == requested)
|
|
||||||
return asm;
|
|
||||||
|
|
||||||
//load missing assemblies by trying to find them in the dll directory
|
//load missing assemblies by trying to find them in the dll directory
|
||||||
string dllname = $"{new AssemblyName(requested).Name}.dll";
|
var dllname = $"{new AssemblyName(requested).Name}.dll";
|
||||||
string directory = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "dll");
|
var directory = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "dll");
|
||||||
string simpleName = new AssemblyName(requested).Name;
|
var simpleName = new AssemblyName(requested).Name;
|
||||||
if (simpleName == "NLua" || simpleName == "KopiLua") directory = Path.Combine(directory, "nlua");
|
if (simpleName == "NLua" || simpleName == "KopiLua") directory = Path.Combine(directory, "nlua");
|
||||||
string fname = Path.Combine(directory, dllname);
|
var 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/unmanaged) assemblies can't load
|
||||||
|
return File.Exists(fname) ? Assembly.LoadFile(fname) : 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
|
#if WINDOWS
|
||||||
public class SingleInstanceController : WindowsFormsApplicationBase
|
private class SingleInstanceController : WindowsFormsApplicationBase
|
||||||
{
|
{
|
||||||
readonly string[] cmdArgs;
|
private readonly string[] cmdArgs;
|
||||||
|
|
||||||
public SingleInstanceController(string[] args)
|
public SingleInstanceController(string[] args)
|
||||||
{
|
{
|
||||||
cmdArgs = args;
|
cmdArgs = args;
|
||||||
|
@ -366,10 +346,12 @@ namespace BizHawk.Client.EmuHawk
|
||||||
StartupNextInstance += this_StartupNextInstance;
|
StartupNextInstance += this_StartupNextInstance;
|
||||||
}
|
}
|
||||||
|
|
||||||
void this_StartupNextInstance(object sender, StartupNextInstanceEventArgs e)
|
public void Run() => Run(cmdArgs);
|
||||||
|
|
||||||
|
private void this_StartupNextInstance(object sender, StartupNextInstanceEventArgs e)
|
||||||
{
|
{
|
||||||
if (e.CommandLine.Count >= 1)
|
if (e.CommandLine.Count >= 1)
|
||||||
(MainForm as MainForm).LoadRom(e.CommandLine[0], new MainForm.LoadRomArgs() { OpenAdvanced = new OpenAdvanced_OpenRom() });
|
((MainForm)MainForm).LoadRom(e.CommandLine[0], new MainForm.LoadRomArgs { OpenAdvanced = new OpenAdvanced_OpenRom() });
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void OnCreateMainForm()
|
protected override void OnCreateMainForm()
|
||||||
|
@ -378,11 +360,9 @@ namespace BizHawk.Client.EmuHawk
|
||||||
var title = MainForm.Text;
|
var title = MainForm.Text;
|
||||||
MainForm.Show();
|
MainForm.Show();
|
||||||
MainForm.Text = title;
|
MainForm.Text = title;
|
||||||
GlobalWin.ExitCode = (MainForm as MainForm).ProgramRunLoop();
|
GlobalWin.ExitCode = ((MainForm)MainForm).ProgramRunLoop();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue