using System; using System.Collections.Generic; using System.ComponentModel; using System.Diagnostics; using System.IO; using System.Linq; using System.Reflection; using System.Runtime.InteropServices; using System.Text; using Dalamud.Game; using Newtonsoft.Json; using Reloaded.Memory.Buffers; using Serilog; using Serilog.Core; using Serilog.Events; using static Dalamud.Injector.NativeFunctions; namespace Dalamud.Injector { /// /// Entrypoint to the program. /// public sealed class EntryPoint { /// /// A delegate used during initialization of the CLR from Dalamud.Injector.Boot. /// /// Count of arguments. /// char** string arguments. public delegate void MainDelegate(int argc, IntPtr argvPtr); /// /// Start the Dalamud injector. /// /// Count of arguments. /// byte** string arguments. public static void Main(int argc, IntPtr argvPtr) { List args = new(argc); unsafe { var argv = (IntPtr*)argvPtr; for (var i = 0; i < argc; i++) args.Add(Marshal.PtrToStringUni(argv[i])); } Init(args); args.Remove("-v"); // Remove "verbose" flag if (args.Count >= 2 && args[1].ToLowerInvariant() == "launch-test") { Environment.Exit(ProcessLaunchTestCommand(args)); return; } // Set boot defaults SetEnvDefaultIfNotExist("DALAMUD_GAMEFIX_LIST", "prevent_devicechange_crashes,disable_game_openprocess_access_check,redirect_openprocess"); SetEnvDefaultIfNotExist("DALAMUD_DOTNET_OPENPROCESS_HOOKMODE", "0"); //SetEnvDefaultIfNotExist("DALAMUD_UNHOOK_DLLS", "kernel32.dll,ntdll.dll,user32.dll"); DalamudStartInfo startInfo = null; if (args.Count == 1) { // No command defaults to inject args.Add("inject"); args.Add("--all"); #if !DEBUG args.Add("--warn"); #endif } else if (int.TryParse(args[1], out var _)) { // Assume that PID has been passed. args.Insert(1, "inject"); // If originally second parameter exists, then assume that it's a base64 encoded start info. // Dalamud.Injector.exe inject [pid] [base64] if (args.Count == 4) { startInfo = JsonConvert.DeserializeObject(Encoding.UTF8.GetString(Convert.FromBase64String(args[3]))); args.RemoveAt(3); } } startInfo = ExtractAndInitializeStartInfoFromArguments(startInfo, args); var mainCommand = args[1].ToLowerInvariant(); if (mainCommand.Length > 0 && mainCommand.Length <= 6 && "inject"[..mainCommand.Length] == mainCommand) { Environment.Exit(ProcessInjectCommand(args, startInfo)); } else if (mainCommand.Length > 0 && mainCommand.Length <= 6 && "launch"[..mainCommand.Length] == mainCommand) { Environment.Exit(ProcessLaunchCommand(args, startInfo)); } else if (mainCommand.Length > 0 && mainCommand.Length <= 4 && "help"[..mainCommand.Length] == mainCommand) { Environment.Exit(ProcessHelpCommand(args, args.Count >= 3 ? args[2] : null)); } else { throw new CommandLineException($"\"{mainCommand}\" is not a valid command."); } } private static void SetEnvDefaultIfNotExist(string name, string value) { var prevValue = Environment.GetEnvironmentVariable(name); if (string.IsNullOrWhiteSpace(prevValue)) { Environment.SetEnvironmentVariable(name, value); } } private static string GetLogPath(string filename) { var baseDirectory = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location); #if DEBUG var logPath = Path.Combine(baseDirectory, $"{filename}.log"); #else var logPath = Path.Combine(baseDirectory, "..", "..", "..", $"{filename}.log"); #endif return logPath; } private static void Init(List args) { InitLogging(args.Any(x => x == "-v")); InitUnhandledException(args); var cwd = new FileInfo(Assembly.GetExecutingAssembly().Location).Directory; if (cwd.FullName != Directory.GetCurrentDirectory()) { Log.Debug($"Changing cwd to {cwd}"); Directory.SetCurrentDirectory(cwd.FullName); } } private static void InitUnhandledException(List args) { AppDomain.CurrentDomain.UnhandledException += (sender, eventArgs) => { var exObj = eventArgs.ExceptionObject; if (exObj is CommandLineException clex) { Console.WriteLine(); Console.WriteLine("Command line error: {0}", clex.Message); Console.WriteLine(); ProcessHelpCommand(args); Environment.Exit(-1); } else if (Log.Logger == null) { Console.WriteLine($"A fatal error has occurred: {eventArgs.ExceptionObject}"); } else if (exObj is Exception ex) { Log.Error(ex, "A fatal error has occurred."); } else { Log.Error($"A fatal error has occurred: {eventArgs.ExceptionObject}"); } #if DEBUG var caption = "Debug Error"; var message = $"Couldn't inject.\nMake sure that Dalamud was not injected into your target process " + $"as a release build before and that the target process can be accessed with VM_WRITE permissions.\n\n" + $"{eventArgs.ExceptionObject}"; #else var caption = "XIVLauncher Error"; var message = "Failed to inject the XIVLauncher in-game addon.\nPlease try restarting your game and your PC.\n" + "If this keeps happening, please report this error."; #endif _ = MessageBoxW(IntPtr.Zero, message, caption, MessageBoxType.IconError | MessageBoxType.Ok); Environment.Exit(-1); }; } private static void InitLogging(bool verbose) { #if DEBUG verbose = true; #endif var levelSwitch = new LoggingLevelSwitch { MinimumLevel = verbose ? LogEventLevel.Verbose : LogEventLevel.Information, }; var logPath = GetLogPath("dalamud.injector"); Environment.SetEnvironmentVariable("DALAMUD_BOOT_LOGFILE", GetLogPath("dalamud.boot")); CullLogFile(logPath, 1 * 1024 * 1024); Log.Logger = new LoggerConfiguration() .WriteTo.Console(standardErrorFromLevel: LogEventLevel.Verbose) .WriteTo.Async(a => a.File(logPath)) .MinimumLevel.ControlledBy(levelSwitch) .CreateLogger(); } private static void CullLogFile(string logPath, int cullingFileSize) { try { var bufferSize = 4096; var logFile = new FileInfo(logPath); if (!logFile.Exists) logFile.Create(); if (logFile.Length <= cullingFileSize) return; var amountToCull = logFile.Length - cullingFileSize; if (amountToCull < bufferSize) return; using var reader = new BinaryReader(logFile.Open(FileMode.Open, FileAccess.Read, FileShare.ReadWrite)); using var writer = new BinaryWriter(logFile.Open(FileMode.Open, FileAccess.Write, FileShare.ReadWrite)); reader.BaseStream.Seek(amountToCull, SeekOrigin.Begin); var read = -1; var total = 0; var buffer = new byte[bufferSize]; while (read != 0) { read = reader.Read(buffer, 0, buffer.Length); writer.Write(buffer, 0, read); total += read; } writer.BaseStream.SetLength(total); } catch (Exception) { /* var caption = "XIVLauncher Error"; var message = $"Log cull threw an exception: {ex.Message}\n{ex.StackTrace ?? string.Empty}"; _ = MessageBoxW(IntPtr.Zero, message, caption, MessageBoxType.IconError | MessageBoxType.Ok); */ } } private static DalamudStartInfo ExtractAndInitializeStartInfoFromArguments(DalamudStartInfo? startInfo, List args) { int len; string key; if (startInfo == null) startInfo = new(); var workingDirectory = startInfo.WorkingDirectory; var configurationPath = startInfo.ConfigurationPath; var pluginDirectory = startInfo.PluginDirectory; var defaultPluginDirectory = startInfo.DefaultPluginDirectory; var assetDirectory = startInfo.AssetDirectory; var delayInitializeMs = startInfo.DelayInitializeMs; var languageStr = startInfo.Language.ToString().ToLowerInvariant(); for (var i = 2; i < args.Count; i++) { if (args[i].StartsWith(key = "--dalamud-working-directory=")) workingDirectory = args[i][key.Length..]; else if (args[i].StartsWith(key = "--dalamud-configuration-path=")) configurationPath = args[i][key.Length..]; else if (args[i].StartsWith(key = "--dalamud-plugin-directory=")) pluginDirectory = args[i][key.Length..]; else if (args[i].StartsWith(key = "--dalamud-dev-plugin-directory=")) defaultPluginDirectory = args[i][key.Length..]; else if (args[i].StartsWith(key = "--dalamud-asset-directory=")) assetDirectory = args[i][key.Length..]; else if (args[i].StartsWith(key = "--dalamud-delay-initialize=")) delayInitializeMs = int.Parse(args[i][key.Length..]); else if (args[i].StartsWith(key = "--dalamud-client-language=")) languageStr = args[i][key.Length..].ToLowerInvariant(); else continue; args.RemoveAt(i); i--; } var appDataDir = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData); var xivlauncherDir = Path.Combine(appDataDir, "XIVLauncher"); workingDirectory ??= Directory.GetCurrentDirectory(); configurationPath ??= Path.Combine(xivlauncherDir, "dalamudConfig.json"); pluginDirectory ??= Path.Combine(xivlauncherDir, "installedPlugins"); defaultPluginDirectory ??= Path.Combine(xivlauncherDir, "devPlugins"); assetDirectory ??= Path.Combine(xivlauncherDir, "dalamudAssets", "dev"); ClientLanguage clientLanguage; if (languageStr[0..(len = Math.Min(languageStr.Length, (key = "english").Length))] == key[0..len]) clientLanguage = ClientLanguage.English; else if (languageStr[0..(len = Math.Min(languageStr.Length, (key = "japanese").Length))] == key[0..len]) clientLanguage = ClientLanguage.Japanese; else if (languageStr[0..(len = Math.Min(languageStr.Length, (key = "日本語").Length))] == key[0..len]) clientLanguage = ClientLanguage.Japanese; else if (languageStr[0..(len = Math.Min(languageStr.Length, (key = "german").Length))] == key[0..len]) clientLanguage = ClientLanguage.German; else if (languageStr[0..(len = Math.Min(languageStr.Length, (key = "deutsche").Length))] == key[0..len]) clientLanguage = ClientLanguage.German; else if (languageStr[0..(len = Math.Min(languageStr.Length, (key = "french").Length))] == key[0..len]) clientLanguage = ClientLanguage.French; else if (languageStr[0..(len = Math.Min(languageStr.Length, (key = "français").Length))] == key[0..len]) clientLanguage = ClientLanguage.French; else if (int.TryParse(languageStr, out var languageInt) && Enum.IsDefined((ClientLanguage)languageInt)) clientLanguage = (ClientLanguage)languageInt; else throw new CommandLineException($"\"{languageStr}\" is not a valid supported language."); return new() { WorkingDirectory = workingDirectory, ConfigurationPath = configurationPath, PluginDirectory = pluginDirectory, DefaultPluginDirectory = defaultPluginDirectory, AssetDirectory = assetDirectory, Language = clientLanguage, GameVersion = null, DelayInitializeMs = delayInitializeMs, }; } private static int ProcessHelpCommand(List args, string? particularCommand = default) { var exeName = Path.GetFileName(args[0]); var exeSpaces = string.Empty; for (var i = exeName.Length; i > 0; i--) exeSpaces += " "; if (particularCommand is null or "help") Console.WriteLine("{0} help [command]", exeName); if (particularCommand is null or "inject") Console.WriteLine("{0} inject [-h/--help] [-a/--all] [--warn] [--fix-acl] [--se-debug-privilege] [pid1] [pid2] [pid3] ...", exeName); if (particularCommand is null or "launch") { Console.WriteLine("{0} launch [-h/--help] [-f/--fake-arguments]", exeName); Console.WriteLine("{0} [-g path/to/ffxiv_dx11.exe] [--game=path/to/ffxiv_dx11.exe]", exeSpaces); Console.WriteLine("{0} [-m entrypoint|inject] [--mode=entrypoint|inject]", exeSpaces); Console.WriteLine("{0} [--handle-owner=inherited-handle-value]", exeSpaces); Console.WriteLine("{0} [--without-dalamud] [--no-fix-acl]", exeSpaces); Console.WriteLine("{0} [-- game_arg1=value1 game_arg2=value2 ...]", exeSpaces); } Console.WriteLine("Specifying dalamud start info: [--dalamud-working-directory=path] [--dalamud-configuration-path=path]"); Console.WriteLine(" [--dalamud-plugin-directory=path] [--dalamud-dev-plugin-directory=path]"); Console.WriteLine(" [--dalamud-asset-directory=path] [--dalamud-delay-initialize=0(ms)]"); Console.WriteLine(" [--dalamud-client-language=0-3|j(apanese)|e(nglish)|d|g(erman)|f(rench)]"); Console.WriteLine("Verbose logging: [-v]"); return 0; } private static int ProcessInjectCommand(List args, DalamudStartInfo dalamudStartInfo) { List processes = new(); var targetProcessSpecified = false; var warnManualInjection = false; var showHelp = args.Count <= 2; var tryFixAcl = false; var tryClaimSeDebugPrivilege = false; for (var i = 2; i < args.Count; i++) { if (int.TryParse(args[i], out int pid)) { targetProcessSpecified = true; try { processes.Add(Process.GetProcessById(pid)); } catch (ArgumentException) { Log.Error("Could not find process with PID: {Pid}", pid); } continue; } if (args[i] == "-h" || args[i] == "--help") { showHelp = true; } else if (args[i] == "-a" || args[i] == "--all") { targetProcessSpecified = true; processes.AddRange(Process.GetProcessesByName("ffxiv_dx11")); } else if (args[i] == "--fix-acl" || args[i] == "--acl-fix") { tryFixAcl = true; } else if (args[i] == "--se-debug-privilege") { tryClaimSeDebugPrivilege = true; } else if (args[i] == "--warn") { warnManualInjection = true; } else { throw new CommandLineException($"\"{args[i]}\" is not a command line argument."); } } if (showHelp) { ProcessHelpCommand(args, "inject"); return args.Count <= 2 ? -1 : 0; } if (!targetProcessSpecified) { throw new CommandLineException("No target process has been specified. Use -a(--all) option to inject to all ffxiv_dx11.exe processes."); } else if (!processes.Any()) { Log.Error("No suitable target process has been found."); return -1; } if (warnManualInjection) { var result = MessageBoxW(IntPtr.Zero, $"Take care: you are manually injecting Dalamud into FFXIV({string.Join(", ", processes.Select(x => $"{x.Id}"))}).\n\nIf you are doing this to use plugins before they are officially whitelisted on patch days, things may go wrong and you may get into trouble.\nWe discourage you from doing this and you won't be warned again in-game.", "Dalamud", MessageBoxType.IconWarning | MessageBoxType.OkCancel); // IDCANCEL if (result == 2) { Log.Information("User cancelled injection"); return -2; } } if (tryClaimSeDebugPrivilege) { try { NativeAclFix.ClaimSeDebug(); Log.Information("SeDebugPrivilege claimed."); } catch (Win32Exception e2) { Log.Warning(e2, "Failed to claim SeDebugPrivilege"); } } foreach (var process in processes) Inject(process, AdjustStartInfo(dalamudStartInfo, process.MainModule.FileName), tryFixAcl); return 0; } private static int ProcessLaunchCommand(List args, DalamudStartInfo dalamudStartInfo) { string? gamePath = null; List gameArguments = new(); string? mode = null; var useFakeArguments = false; var showHelp = args.Count <= 2; var handleOwner = IntPtr.Zero; var withoutDalamud = false; var noFixAcl = false; var parsingGameArgument = false; for (var i = 2; i < args.Count; i++) { if (parsingGameArgument) { gameArguments.Add(args[i]); continue; } if (args[i] == "-h" || args[i] == "--help") showHelp = true; else if (args[i] == "-f" || args[i] == "--fake-arguments") useFakeArguments = true; else if (args[i] == "--without-dalamud") withoutDalamud = true; else if (args[i] == "--no-fix-acl" || args[i] == "--no-acl-fix") noFixAcl = true; else if (args[i] == "-g") gamePath = args[++i]; else if (args[i].StartsWith("--game=")) gamePath = args[i].Split('=', 2)[1]; else if (args[i] == "-m") mode = args[++i]; else if (args[i].StartsWith("--mode=")) mode = args[i].Split('=', 2)[1]; else if (args[i].StartsWith("--handle-owner=")) handleOwner = IntPtr.Parse(args[i].Split('=', 2)[1]); else if (args[i] == "--") parsingGameArgument = true; else throw new CommandLineException($"\"{args[i]}\" is not a command line argument."); } if (showHelp) { ProcessHelpCommand(args, "launch"); return args.Count <= 2 ? -1 : 0; } mode = mode == null ? "entrypoint" : mode.ToLowerInvariant(); if (mode.Length > 0 && mode.Length <= 10 && "entrypoint"[0..mode.Length] == mode) { mode = "entrypoint"; } else if (mode.Length > 0 && mode.Length <= 6 && "inject"[0..mode.Length] == mode) { mode = "inject"; } else { throw new CommandLineException($"\"{mode}\" is not a valid Dalamud load mode."); } if (gamePath == null) { try { var appDataDir = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData); var xivlauncherDir = Path.Combine(appDataDir, "XIVLauncher"); var launcherConfigPath = Path.Combine(xivlauncherDir, "launcherConfigV3.json"); gamePath = Path.Combine(JsonSerializer.CreateDefault().Deserialize>(new JsonTextReader(new StringReader(File.ReadAllText(launcherConfigPath))))["GamePath"], "game", "ffxiv_dx11.exe"); Log.Information("Using game installation path configuration from from XIVLauncher: {0}", gamePath); } catch (Exception) { gamePath = @"C:\Program Files (x86)\SquareEnix\FINAL FANTASY XIV - A Realm Reborn\game\ffxiv_dx11.exe"; Log.Warning("Failed to read launcherConfigV3.json. Using default game installation path: {0}", gamePath); } if (!File.Exists(gamePath)) { Log.Error("File not found: {0}", gamePath); return -1; } } if (useFakeArguments) { var gameVersion = File.ReadAllText(Path.Combine(Directory.GetParent(gamePath).FullName, "ffxivgame.ver")); var sqpackPath = Path.Combine(Directory.GetParent(gamePath).FullName, "sqpack"); var maxEntitledExpansionId = 0; while (File.Exists(Path.Combine(sqpackPath, $"ex{maxEntitledExpansionId + 1}", $"ex{maxEntitledExpansionId + 1}.ver"))) maxEntitledExpansionId++; gameArguments.InsertRange(0, new string[] { "DEV.TestSID=0", "DEV.UseSqPack=1", "DEV.DataPathType=1", "DEV.LobbyHost01=127.0.0.1", "DEV.LobbyPort01=54994", "DEV.LobbyHost02=127.0.0.2", "DEV.LobbyPort02=54994", "DEV.LobbyHost03=127.0.0.3", "DEV.LobbyPort03=54994", "DEV.LobbyHost04=127.0.0.4", "DEV.LobbyPort04=54994", "DEV.LobbyHost05=127.0.0.5", "DEV.LobbyPort05=54994", "DEV.LobbyHost06=127.0.0.6", "DEV.LobbyPort06=54994", "DEV.LobbyHost07=127.0.0.7", "DEV.LobbyPort07=54994", "DEV.LobbyHost08=127.0.0.8", "DEV.LobbyPort08=54994", "DEV.LobbyHost09=127.0.0.9", "DEV.LobbyPort09=54994", "SYS.Region=0", $"language={(int)dalamudStartInfo.Language}", $"ver={gameVersion}", $"DEV.MaxEntitledExpansionID={maxEntitledExpansionId}", "DEV.GMServerHost=127.0.0.100", "DEV.GameQuitMessageBox=0", }); } var gameArgumentString = string.Join(" ", gameArguments.Select(x => EncodeParameterArgument(x))); var process = NativeAclFix.LaunchGame(Path.GetDirectoryName(gamePath), gamePath, gameArgumentString, noFixAcl, (Process p) => { if (!withoutDalamud && mode == "entrypoint") { var startInfo = AdjustStartInfo(dalamudStartInfo, gamePath); Log.Information("Using start info: {0}", JsonConvert.SerializeObject(startInfo)); if (RewriteRemoteEntryPointW(p.Handle, gamePath, JsonConvert.SerializeObject(startInfo)) != 0) { Log.Error("[HOOKS] RewriteRemoteEntryPointW failed"); throw new Exception("RewriteRemoteEntryPointW failed"); } } }); if (!withoutDalamud && mode == "inject") { var startInfo = AdjustStartInfo(dalamudStartInfo, gamePath); Log.Information("Using start info: {0}", JsonConvert.SerializeObject(startInfo)); Inject(process, startInfo, false); } var processHandleForOwner = IntPtr.Zero; if (handleOwner != IntPtr.Zero) { if (!DuplicateHandle(Process.GetCurrentProcess().Handle, process.Handle, handleOwner, out processHandleForOwner, 0, false, DuplicateOptions.SameAccess)) Log.Warning("Failed to call DuplicateHandle: Win32 error code {0}", Marshal.GetLastWin32Error()); } Console.WriteLine($"{{\"pid\": {process.Id}, \"handle\": {processHandleForOwner}}}"); return 0; } private static Process GetInheritableCurrentProcessHandle() { if (!DuplicateHandle(Process.GetCurrentProcess().Handle, Process.GetCurrentProcess().Handle, Process.GetCurrentProcess().Handle, out var inheritableCurrentProcessHandle, 0, true, DuplicateOptions.SameAccess)) { Log.Error("Failed to call DuplicateHandle: Win32 error code {0}", Marshal.GetLastWin32Error()); return null; } return new ExistingProcess(inheritableCurrentProcessHandle); } private static int ProcessLaunchTestCommand(List args) { Console.WriteLine("Testing launch command."); args[0] = Process.GetCurrentProcess().MainModule.FileName; args[1] = "launch"; var inheritableCurrentProcess = GetInheritableCurrentProcessHandle(); // so that it closes the handle when it's done args.Insert(2, $"--handle-owner={inheritableCurrentProcess.Handle}"); for (var i = 0; i < args.Count; i++) Console.WriteLine("Argument {0}: {1}", i, args[i]); Process helperProcess = new(); helperProcess.StartInfo.FileName = args[0]; for (var i = 1; i < args.Count; i++) helperProcess.StartInfo.ArgumentList.Add(args[i]); helperProcess.StartInfo.RedirectStandardOutput = true; helperProcess.StartInfo.RedirectStandardError = true; helperProcess.StartInfo.UseShellExecute = false; helperProcess.ErrorDataReceived += new DataReceivedEventHandler((sendingProcess, errLine) => Console.WriteLine($"stderr: \"{errLine.Data}\"")); helperProcess.Start(); helperProcess.BeginErrorReadLine(); helperProcess.WaitForExit(); if (helperProcess.ExitCode != 0) return -1; var result = JsonSerializer.CreateDefault().Deserialize>(new JsonTextReader(helperProcess.StandardOutput)); var pid = result["pid"]; var handle = (IntPtr)result["handle"]; var resultProcess = new ExistingProcess(handle); Console.WriteLine("PID: {0}, Handle: {1}", pid, handle); Console.WriteLine("Press Enter to force quit"); Console.ReadLine(); resultProcess.Kill(); return 0; } private static DalamudStartInfo AdjustStartInfo(DalamudStartInfo startInfo, string gamePath) { var ffxivDir = Path.GetDirectoryName(gamePath); var gameVerStr = File.ReadAllText(Path.Combine(ffxivDir, "ffxivgame.ver")); var gameVer = GameVersion.Parse(gameVerStr); return new() { WorkingDirectory = startInfo.WorkingDirectory, ConfigurationPath = startInfo.ConfigurationPath, PluginDirectory = startInfo.PluginDirectory, DefaultPluginDirectory = startInfo.DefaultPluginDirectory, AssetDirectory = startInfo.AssetDirectory, Language = startInfo.Language, GameVersion = gameVer, DelayInitializeMs = startInfo.DelayInitializeMs, }; } private static void Inject(Process process, DalamudStartInfo startInfo, bool tryFixAcl = false) { if (tryFixAcl) { try { NativeAclFix.CopyAclFromSelfToTargetProcess(process.SafeHandle.DangerousGetHandle()); } catch (Win32Exception e1) { Log.Warning(e1, "Failed to copy ACL"); } } var bootName = "Dalamud.Boot.dll"; var bootPath = Path.GetFullPath(bootName); // ====================================================== using var injector = new Injector(process, false); injector.LoadLibrary(bootPath, out var bootModule); // ====================================================== var startInfoJson = JsonConvert.SerializeObject(startInfo); var startInfoBytes = Encoding.UTF8.GetBytes(startInfoJson); using var startInfoBuffer = new MemoryBufferHelper(process).CreatePrivateMemoryBuffer(startInfoBytes.Length + 0x8); var startInfoAddress = startInfoBuffer.Add(startInfoBytes); if (startInfoAddress == IntPtr.Zero) throw new Exception("Unable to allocate start info JSON"); injector.GetFunctionAddress(bootModule, "Initialize", out var initAddress); injector.CallRemoteFunction(initAddress, startInfoAddress, out var exitCode); // ====================================================== if (exitCode > 0) { Log.Error($"Dalamud.Boot::Initialize returned {exitCode}"); return; } Log.Information("Done"); } [DllImport("Dalamud.Boot.dll")] private static extern int RewriteRemoteEntryPointW(IntPtr hProcess, [MarshalAs(UnmanagedType.LPWStr)] string gamePath, [MarshalAs(UnmanagedType.LPWStr)] string loadInfoJson); /// /// This routine appends the given argument to a command line such that /// CommandLineToArgvW will return the argument string unchanged. Arguments /// in a command line should be separated by spaces; this function does /// not add these spaces. /// /// Taken from https://stackoverflow.com/questions/5510343/escape-command-line-arguments-in-c-sharp /// and https://blogs.msdn.microsoft.com/twistylittlepassagesallalike/2011/04/23/everyone-quotes-command-line-arguments-the-wrong-way/. /// /// Supplies the argument to encode. /// /// Supplies an indication of whether we should quote the argument even if it /// does not contain any characters that would ordinarily require quoting. /// private static string EncodeParameterArgument(string argument, bool force = false) { if (argument == null) throw new ArgumentNullException(nameof(argument)); // Unless we're told otherwise, don't quote unless we actually // need to do so --- hopefully avoid problems if programs won't // parse quotes properly if (force == false && argument.Length > 0 && argument.IndexOfAny(" \t\n\v\"".ToCharArray()) == -1) { return argument; } var quoted = new StringBuilder(); quoted.Append('"'); var numberBackslashes = 0; foreach (var chr in argument) { switch (chr) { case '\\': numberBackslashes++; continue; case '"': // Escape all backslashes and the following // double quotation mark. quoted.Append('\\', (numberBackslashes * 2) + 1); quoted.Append(chr); break; default: // Backslashes aren't special here. quoted.Append('\\', numberBackslashes); quoted.Append(chr); break; } numberBackslashes = 0; } // Escape all backslashes, but let the terminating // double quotation mark we add below be interpreted // as a metacharacter. quoted.Append('\\', numberBackslashes * 2); quoted.Append('"'); return quoted.ToString(); } private class CommandLineException : Exception { public CommandLineException(string cause) : base(cause) { } } } }