mirror of
https://github.com/Ryujinx/Ryujinx.git
synced 2025-01-19 10:30:13 +00:00
Add the TamperMachine module for runtime mods and cheats (#1928)
* Add initial implementation of the Tamper Machine * Implement Atmosphere opcodes 0, 4 and 9 * Add missing TamperCompilationException class * Implement Atmosphere conditional and loop opcodes 1, 2 and 3 * Inplement input conditional opcode 8 * Add register store opcode A * Implement extended pause/resume opcodes FF0 and FF1 * Implement extended log opcode FFF * Implement extended register conditional opcode C0 * Refactor TamperProgram to an interface * Moved Atmosphere classes to a separate subdirectory * Fix OpProcCtrl class not setting process * Implement extended register save/restore opcodes C1, C2 and C3 * Refactor code emitters to separate classes * Supress memory access errors from the Tamper Machine * Add debug information to tamper register and memory writes * Add block stack check to Atmosphere Cheat compiler * Add handheld input support to Tamper Machine * Fix code styling * Fix build id and cheat case mismatch * Fix invalid immediate size selection * Print build ids of the title * Prevent Tamper Machine from change code regions * Remove Atmosphere namespace * Remove empty cheats from the list * Prevent code modification without disabling the tampering * Fix missing addressing mode in LoadRegisterWithMemory * Fix wrong addressing in RegisterConditional * Add name to the tamper machine thread * Fix code styling
This commit is contained in:
parent
a5d5ca0635
commit
0c1ea1212a
71 changed files with 2793 additions and 5 deletions
|
@ -57,6 +57,7 @@ namespace Ryujinx.Common.Logging
|
|||
ServiceTime,
|
||||
ServiceVi,
|
||||
SurfaceFlinger,
|
||||
TamperMachine,
|
||||
Vic
|
||||
}
|
||||
}
|
9
Ryujinx.HLE/Exceptions/CodeRegionTamperedException.cs
Normal file
9
Ryujinx.HLE/Exceptions/CodeRegionTamperedException.cs
Normal file
|
@ -0,0 +1,9 @@
|
|||
using System;
|
||||
|
||||
namespace Ryujinx.HLE.Exceptions
|
||||
{
|
||||
public class CodeRegionTamperedException : TamperExecutionException
|
||||
{
|
||||
public CodeRegionTamperedException(string message) : base(message) { }
|
||||
}
|
||||
}
|
9
Ryujinx.HLE/Exceptions/TamperCompilationException.cs
Normal file
9
Ryujinx.HLE/Exceptions/TamperCompilationException.cs
Normal file
|
@ -0,0 +1,9 @@
|
|||
using System;
|
||||
|
||||
namespace Ryujinx.HLE.Exceptions
|
||||
{
|
||||
public class TamperCompilationException : Exception
|
||||
{
|
||||
public TamperCompilationException(string message) : base(message) { }
|
||||
}
|
||||
}
|
9
Ryujinx.HLE/Exceptions/TamperExecutionException.cs
Normal file
9
Ryujinx.HLE/Exceptions/TamperExecutionException.cs
Normal file
|
@ -0,0 +1,9 @@
|
|||
using System;
|
||||
|
||||
namespace Ryujinx.HLE.Exceptions
|
||||
{
|
||||
public class TamperExecutionException : Exception
|
||||
{
|
||||
public TamperExecutionException(string message) : base(message) { }
|
||||
}
|
||||
}
|
|
@ -11,6 +11,7 @@ using Ryujinx.Common.Configuration;
|
|||
using Ryujinx.Common.Logging;
|
||||
using Ryujinx.HLE.FileSystem;
|
||||
using Ryujinx.HLE.FileSystem.Content;
|
||||
using Ryujinx.HLE.HOS.Kernel.Process;
|
||||
using Ryujinx.HLE.Loaders.Executables;
|
||||
using Ryujinx.HLE.Loaders.Npdm;
|
||||
using System;
|
||||
|
@ -527,7 +528,9 @@ namespace Ryujinx.HLE.HOS
|
|||
|
||||
Ptc.Initialize(TitleIdText, DisplayVersion, usePtc);
|
||||
|
||||
ProgramLoader.LoadNsos(_device.System.KernelContext, metaData, executables: programs);
|
||||
ProgramLoader.LoadNsos(_device.System.KernelContext, out ProcessTamperInfo tamperInfo, metaData, executables: programs);
|
||||
|
||||
_fileSystem.ModLoader.LoadCheats(TitleId, tamperInfo, _device.TamperMachine);
|
||||
}
|
||||
|
||||
public void LoadProgram(string filePath)
|
||||
|
@ -626,7 +629,9 @@ namespace Ryujinx.HLE.HOS
|
|||
Graphics.Gpu.GraphicsConfig.TitleId = null;
|
||||
_device.Gpu.HostInitalized.Set();
|
||||
|
||||
ProgramLoader.LoadNsos(_device.System.KernelContext, metaData, executables: executable);
|
||||
ProgramLoader.LoadNsos(_device.System.KernelContext, out ProcessTamperInfo tamperInfo, metaData, executables: executable);
|
||||
|
||||
_fileSystem.ModLoader.LoadCheats(TitleId, tamperInfo, _device.TamperMachine);
|
||||
}
|
||||
|
||||
private Npdm GetDefaultNpdm()
|
||||
|
|
20
Ryujinx.HLE/HOS/Kernel/Process/ProcessTamperInfo.cs
Normal file
20
Ryujinx.HLE/HOS/Kernel/Process/ProcessTamperInfo.cs
Normal file
|
@ -0,0 +1,20 @@
|
|||
using System.Collections.Generic;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Kernel.Process
|
||||
{
|
||||
internal class ProcessTamperInfo
|
||||
{
|
||||
public KProcess Process { get; }
|
||||
public IEnumerable<string> BuildIds { get; }
|
||||
public IEnumerable<ulong> CodeAddresses { get; }
|
||||
public ulong HeapAddress { get; }
|
||||
|
||||
public ProcessTamperInfo(KProcess process, IEnumerable<string> buildIds, IEnumerable<ulong> codeAddresses, ulong heapAddress)
|
||||
{
|
||||
Process = process;
|
||||
BuildIds = buildIds;
|
||||
CodeAddresses = codeAddresses;
|
||||
HeapAddress = heapAddress;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -13,6 +13,8 @@ using System.Collections.Specialized;
|
|||
using System.Linq;
|
||||
using System.IO;
|
||||
using Ryujinx.HLE.Loaders.Npdm;
|
||||
using Ryujinx.HLE.HOS.Kernel.Process;
|
||||
using System.Globalization;
|
||||
|
||||
namespace Ryujinx.HLE.HOS
|
||||
{
|
||||
|
@ -20,9 +22,12 @@ namespace Ryujinx.HLE.HOS
|
|||
{
|
||||
private const string RomfsDir = "romfs";
|
||||
private const string ExefsDir = "exefs";
|
||||
private const string CheatDir = "cheats";
|
||||
private const string RomfsContainer = "romfs.bin";
|
||||
private const string ExefsContainer = "exefs.nsp";
|
||||
private const string StubExtension = ".stub";
|
||||
private const string CheatExtension = ".txt";
|
||||
private const string DefaultCheatName = "<default>";
|
||||
|
||||
private const string AmsContentsDir = "contents";
|
||||
private const string AmsNsoPatchDir = "exefs_patches";
|
||||
|
@ -41,6 +46,24 @@ namespace Ryujinx.HLE.HOS
|
|||
}
|
||||
}
|
||||
|
||||
public struct Cheat
|
||||
{
|
||||
// Atmosphere identifies the executables with the first 8 bytes
|
||||
// of the build id, which is equivalent to 16 hex digits.
|
||||
public const int CheatIdSize = 16;
|
||||
|
||||
public readonly string Name;
|
||||
public readonly FileInfo Path;
|
||||
public readonly IEnumerable<String> Instructions;
|
||||
|
||||
public Cheat(string name, FileInfo path, IEnumerable<String> instructions)
|
||||
{
|
||||
Name = name;
|
||||
Path = path;
|
||||
Instructions = instructions;
|
||||
}
|
||||
}
|
||||
|
||||
// Title dependent mods
|
||||
public class ModCache
|
||||
{
|
||||
|
@ -50,12 +73,15 @@ namespace Ryujinx.HLE.HOS
|
|||
public List<Mod<DirectoryInfo>> RomfsDirs { get; }
|
||||
public List<Mod<DirectoryInfo>> ExefsDirs { get; }
|
||||
|
||||
public List<Cheat> Cheats { get; }
|
||||
|
||||
public ModCache()
|
||||
{
|
||||
RomfsContainers = new List<Mod<FileInfo>>();
|
||||
ExefsContainers = new List<Mod<FileInfo>>();
|
||||
RomfsDirs = new List<Mod<DirectoryInfo>>();
|
||||
ExefsDirs = new List<Mod<DirectoryInfo>>();
|
||||
Cheats = new List<Cheat>();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -192,20 +218,38 @@ namespace Ryujinx.HLE.HOS
|
|||
mods.ExefsDirs.Add(mod = new Mod<DirectoryInfo>($"<{titleDir.Name} ExeFs>", modDir));
|
||||
types.Append('E');
|
||||
}
|
||||
else if (StrEquals(CheatDir, modDir.Name))
|
||||
{
|
||||
for (int i = 0; i < QueryCheatsDir(mods, modDir); i++)
|
||||
{
|
||||
types.Append('C');
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
var romfs = new DirectoryInfo(Path.Combine(modDir.FullName, RomfsDir));
|
||||
var exefs = new DirectoryInfo(Path.Combine(modDir.FullName, ExefsDir));
|
||||
var cheat = new DirectoryInfo(Path.Combine(modDir.FullName, CheatDir));
|
||||
|
||||
if (romfs.Exists)
|
||||
{
|
||||
mods.RomfsDirs.Add(mod = new Mod<DirectoryInfo>(modDir.Name, romfs));
|
||||
types.Append('R');
|
||||
}
|
||||
|
||||
if (exefs.Exists)
|
||||
{
|
||||
mods.ExefsDirs.Add(mod = new Mod<DirectoryInfo>(modDir.Name, exefs));
|
||||
types.Append('E');
|
||||
}
|
||||
|
||||
if (cheat.Exists)
|
||||
{
|
||||
for (int i = 0; i < QueryCheatsDir(mods, cheat); i++)
|
||||
{
|
||||
types.Append('C');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (types.Length > 0) Logger.Info?.Print(LogClass.ModLoader, $"Found mod '{mod.Name}' [{types}]");
|
||||
|
@ -226,6 +270,94 @@ namespace Ryujinx.HLE.HOS
|
|||
}
|
||||
}
|
||||
|
||||
private static int QueryCheatsDir(ModCache mods, DirectoryInfo cheatsDir)
|
||||
{
|
||||
if (!cheatsDir.Exists)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
int numMods = 0;
|
||||
|
||||
foreach (FileInfo file in cheatsDir.EnumerateFiles())
|
||||
{
|
||||
if (!StrEquals(CheatExtension, file.Extension))
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
string cheatId = Path.GetFileNameWithoutExtension(file.Name);
|
||||
|
||||
if (cheatId.Length != Cheat.CheatIdSize)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
if (!ulong.TryParse(cheatId, NumberStyles.HexNumber, CultureInfo.InvariantCulture, out _))
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
// A cheat file can contain several cheats for the same executable, so the file must be parsed in
|
||||
// order to properly enumerate them.
|
||||
mods.Cheats.AddRange(GetCheatsInFile(file));
|
||||
}
|
||||
|
||||
return numMods;
|
||||
}
|
||||
|
||||
private static IEnumerable<Cheat> GetCheatsInFile(FileInfo cheatFile)
|
||||
{
|
||||
string cheatName = DefaultCheatName;
|
||||
List<string> instructions = new List<string>();
|
||||
List<Cheat> cheats = new List<Cheat>();
|
||||
|
||||
using (StreamReader cheatData = cheatFile.OpenText())
|
||||
{
|
||||
string line;
|
||||
while ((line = cheatData.ReadLine()) != null)
|
||||
{
|
||||
line = line.Trim();
|
||||
|
||||
if (line.StartsWith('['))
|
||||
{
|
||||
// This line starts a new cheat section.
|
||||
if (!line.EndsWith(']') || line.Length < 3)
|
||||
{
|
||||
// Skip the entire file if there's any error while parsing the cheat file.
|
||||
|
||||
Logger.Warning?.Print(LogClass.ModLoader, $"Ignoring cheat '{cheatFile.FullName}' because it is malformed");
|
||||
|
||||
return new List<Cheat>();
|
||||
}
|
||||
|
||||
// Add the previous section to the list.
|
||||
if (instructions.Count != 0)
|
||||
{
|
||||
cheats.Add(new Cheat($"<{cheatName} Cheat>", cheatFile, instructions));
|
||||
}
|
||||
|
||||
// Start a new cheat section.
|
||||
cheatName = line.Substring(1, line.Length - 2);
|
||||
instructions = new List<string>();
|
||||
}
|
||||
else if (line.Length > 0)
|
||||
{
|
||||
// The line contains an instruction.
|
||||
instructions.Add(line);
|
||||
}
|
||||
}
|
||||
|
||||
// Add the last section being processed.
|
||||
if (instructions.Count != 0)
|
||||
{
|
||||
cheats.Add(new Cheat($"<{cheatName} Cheat>", cheatFile, instructions));
|
||||
}
|
||||
}
|
||||
|
||||
return cheats;
|
||||
}
|
||||
|
||||
// Assumes searchDirPaths don't overlap
|
||||
public static void CollectMods(Dictionary<ulong, ModCache> modCaches, PatchCache patches, params string[] searchDirPaths)
|
||||
{
|
||||
|
@ -408,7 +540,6 @@ namespace Ryujinx.HLE.HOS
|
|||
return modLoadResult;
|
||||
}
|
||||
|
||||
|
||||
if (nsos.Length != ApplicationLoader.ExeFsPrefixes.Length)
|
||||
{
|
||||
throw new ArgumentOutOfRangeException("NSO Count is incorrect");
|
||||
|
@ -494,6 +625,41 @@ namespace Ryujinx.HLE.HOS
|
|||
return ApplyProgramPatches(nsoMods, 0x100, programs);
|
||||
}
|
||||
|
||||
internal void LoadCheats(ulong titleId, ProcessTamperInfo tamperInfo, TamperMachine tamperMachine)
|
||||
{
|
||||
if (tamperInfo == null || tamperInfo.BuildIds == null || tamperInfo.CodeAddresses == null)
|
||||
{
|
||||
Logger.Error?.Print(LogClass.ModLoader, "Unable to install cheat because the associated process is invalid");
|
||||
}
|
||||
|
||||
Logger.Info?.Print(LogClass.ModLoader, $"Build ids found for title {titleId:X16}:\n {String.Join("\n ", tamperInfo.BuildIds)}");
|
||||
|
||||
if (!AppMods.TryGetValue(titleId, out ModCache mods) || mods.Cheats.Count == 0)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
var cheats = mods.Cheats;
|
||||
var processExes = tamperInfo.BuildIds.Zip(tamperInfo.CodeAddresses, (k, v) => new { k, v })
|
||||
.ToDictionary(x => x.k.Substring(0, Math.Min(Cheat.CheatIdSize, x.k.Length)), x => x.v);
|
||||
|
||||
foreach (var cheat in cheats)
|
||||
{
|
||||
string cheatId = Path.GetFileNameWithoutExtension(cheat.Path.Name).ToUpper();
|
||||
|
||||
if (!processExes.TryGetValue(cheatId, out ulong exeAddress))
|
||||
{
|
||||
Logger.Warning?.Print(LogClass.ModLoader, $"Skipping cheat '{cheat.Name}' because no executable matches its BuildId {cheatId} (check if the game title and version are correct)");
|
||||
|
||||
continue;
|
||||
}
|
||||
|
||||
Logger.Info?.Print(LogClass.ModLoader, $"Installing cheat '{cheat.Name}'");
|
||||
|
||||
tamperMachine.InstallAtmosphereCheat(cheat.Instructions, tamperInfo, exeAddress);
|
||||
}
|
||||
}
|
||||
|
||||
private static bool ApplyProgramPatches(IEnumerable<Mod<DirectoryInfo>> mods, int protectedOffset, params IExecutable[] programs)
|
||||
{
|
||||
int count = 0;
|
||||
|
|
|
@ -1,13 +1,14 @@
|
|||
using ARMeilleure.Translation.PTC;
|
||||
using Ryujinx.Common;
|
||||
using Ryujinx.Common.Logging;
|
||||
using Ryujinx.Cpu;
|
||||
using Ryujinx.HLE.HOS.Kernel;
|
||||
using Ryujinx.HLE.HOS.Kernel.Common;
|
||||
using Ryujinx.HLE.HOS.Kernel.Memory;
|
||||
using Ryujinx.HLE.HOS.Kernel.Process;
|
||||
using Ryujinx.HLE.Loaders.Executables;
|
||||
using Ryujinx.HLE.Loaders.Npdm;
|
||||
using System;
|
||||
using System.Linq;
|
||||
|
||||
namespace Ryujinx.HLE.HOS
|
||||
{
|
||||
|
@ -124,13 +125,20 @@ namespace Ryujinx.HLE.HOS
|
|||
return true;
|
||||
}
|
||||
|
||||
public static bool LoadNsos(KernelContext context, Npdm metaData, byte[] arguments = null, params IExecutable[] executables)
|
||||
public static bool LoadNsos(KernelContext context, out ProcessTamperInfo tamperInfo, Npdm metaData, byte[] arguments = null, params IExecutable[] executables)
|
||||
{
|
||||
ulong argsStart = 0;
|
||||
uint argsSize = 0;
|
||||
ulong codeStart = metaData.Is64Bit ? 0x8000000UL : 0x200000UL;
|
||||
uint codeSize = 0;
|
||||
|
||||
var buildIds = executables.Select(e => (e switch
|
||||
{
|
||||
NsoExecutable nso => BitConverter.ToString(nso.BuildId.Bytes.ToArray()),
|
||||
NroExecutable nro => BitConverter.ToString(nro.Header.BuildId),
|
||||
_ => ""
|
||||
}).Replace("-", "").ToUpper());
|
||||
|
||||
ulong[] nsoBase = new ulong[executables.Length];
|
||||
|
||||
for (int index = 0; index < executables.Length; index++)
|
||||
|
@ -202,6 +210,8 @@ namespace Ryujinx.HLE.HOS
|
|||
{
|
||||
Logger.Error?.Print(LogClass.Loader, $"Process initialization failed setting resource limit values.");
|
||||
|
||||
tamperInfo = null;
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
|
@ -213,6 +223,8 @@ namespace Ryujinx.HLE.HOS
|
|||
{
|
||||
Logger.Error?.Print(LogClass.Loader, $"Process initialization failed due to invalid ACID flags.");
|
||||
|
||||
tamperInfo = null;
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
|
@ -229,6 +241,8 @@ namespace Ryujinx.HLE.HOS
|
|||
{
|
||||
Logger.Error?.Print(LogClass.Loader, $"Process initialization returned error \"{result}\".");
|
||||
|
||||
tamperInfo = null;
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
|
@ -242,6 +256,8 @@ namespace Ryujinx.HLE.HOS
|
|||
{
|
||||
Logger.Error?.Print(LogClass.Loader, $"Process initialization returned error \"{result}\".");
|
||||
|
||||
tamperInfo = null;
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
@ -254,11 +270,18 @@ namespace Ryujinx.HLE.HOS
|
|||
{
|
||||
Logger.Error?.Print(LogClass.Loader, $"Process start returned error \"{result}\".");
|
||||
|
||||
tamperInfo = null;
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
context.Processes.TryAdd(process.Pid, process);
|
||||
|
||||
// Keep the build ids because the tamper machine uses them to know which process to associate a
|
||||
// tamper to and also keep the starting address of each executable inside a process because some
|
||||
// memory modifications are relative to this address.
|
||||
tamperInfo = new ProcessTamperInfo(process, buildIds, nsoBase, process.MemoryManager.HeapRegionStart);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
|
130
Ryujinx.HLE/HOS/Tamper/AtmosphereCompiler.cs
Normal file
130
Ryujinx.HLE/HOS/Tamper/AtmosphereCompiler.cs
Normal file
|
@ -0,0 +1,130 @@
|
|||
using Ryujinx.Common.Logging;
|
||||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.CodeEmitters;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
class AtmosphereCompiler
|
||||
{
|
||||
public ITamperProgram Compile(IEnumerable<string> rawInstructions, ulong exeAddress, ulong heapAddress, ITamperedProcess process)
|
||||
{
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, $"Executable address: {exeAddress:X16}");
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, $"Heap address: {heapAddress:X16}");
|
||||
|
||||
try
|
||||
{
|
||||
return CompileImpl(rawInstructions, exeAddress, heapAddress, process);
|
||||
}
|
||||
catch(TamperCompilationException exception)
|
||||
{
|
||||
// Just print the message without the stack trace.
|
||||
Logger.Error?.Print(LogClass.TamperMachine, exception.Message);
|
||||
}
|
||||
catch (Exception exception)
|
||||
{
|
||||
Logger.Error?.Print(LogClass.TamperMachine, exception.ToString());
|
||||
}
|
||||
|
||||
Logger.Error?.Print(LogClass.TamperMachine, "There was a problem while compiling the Atmosphere cheat");
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
private ITamperProgram CompileImpl(IEnumerable<string> rawInstructions, ulong exeAddress, ulong heapAddress, ITamperedProcess process)
|
||||
{
|
||||
CompilationContext context = new CompilationContext(exeAddress, heapAddress, process);
|
||||
context.BlockStack.Push(new OperationBlock(null));
|
||||
|
||||
// Parse the instructions.
|
||||
|
||||
foreach (string rawInstruction in rawInstructions)
|
||||
{
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, $"Compiling instruction {rawInstruction}");
|
||||
|
||||
byte[] instruction = InstructionHelper.ParseRawInstruction(rawInstruction);
|
||||
CodeType codeType = InstructionHelper.GetCodeType(instruction);
|
||||
|
||||
switch (codeType)
|
||||
{
|
||||
case CodeType.StoreConstantToAddress:
|
||||
StoreConstantToAddress.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.BeginMemoryConditionalBlock:
|
||||
BeginConditionalBlock.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.EndConditionalBlock:
|
||||
EndConditionalBlock.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.StartEndLoop:
|
||||
StartEndLoop.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.LoadRegisterWithContant:
|
||||
LoadRegisterWithConstant.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.LoadRegisterWithMemory:
|
||||
LoadRegisterWithMemory.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.StoreConstantToMemory:
|
||||
StoreConstantToMemory.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.LegacyArithmetic:
|
||||
LegacyArithmetic.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.BeginKeypressConditionalBlock:
|
||||
BeginConditionalBlock.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.Arithmetic:
|
||||
Arithmetic.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.StoreRegisterToMemory:
|
||||
StoreRegisterToMemory.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.BeginRegisterConditionalBlock:
|
||||
BeginConditionalBlock.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.SaveOrRestoreRegister:
|
||||
SaveOrRestoreRegister.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.SaveOrRestoreRegisterWithMask:
|
||||
SaveOrRestoreRegisterWithMask.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.ReadOrWriteStaticRegister:
|
||||
ReadOrWriteStaticRegister.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.PauseProcess:
|
||||
PauseProcess.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.ResumeProcess:
|
||||
ResumeProcess.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.DebugLog:
|
||||
DebugLog.Emit(instruction, context);
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Code type {codeType} not implemented in Atmosphere cheat");
|
||||
}
|
||||
}
|
||||
|
||||
// Initialize only the registers used.
|
||||
|
||||
Value<ulong> zero = new Value<ulong>(0UL);
|
||||
int position = 0;
|
||||
|
||||
foreach (Register register in context.Registers.Values)
|
||||
{
|
||||
context.CurrentOperations.Insert(position, new OpMov<ulong>(register, zero));
|
||||
position++;
|
||||
}
|
||||
|
||||
if (context.BlockStack.Count != 1)
|
||||
{
|
||||
throw new TamperCompilationException($"Reached end of compilation with unmatched conditional(s) or loop(s)");
|
||||
}
|
||||
|
||||
return new AtmosphereProgram(process, context.PressedKeys, new Block(context.CurrentOperations));
|
||||
}
|
||||
}
|
||||
}
|
26
Ryujinx.HLE/HOS/Tamper/AtmosphereProgram.cs
Normal file
26
Ryujinx.HLE/HOS/Tamper/AtmosphereProgram.cs
Normal file
|
@ -0,0 +1,26 @@
|
|||
using Ryujinx.HLE.HOS.Services.Hid;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
class AtmosphereProgram : ITamperProgram
|
||||
{
|
||||
private Parameter<long> _pressedKeys;
|
||||
private IOperation _entryPoint;
|
||||
|
||||
public ITamperedProcess Process { get; }
|
||||
|
||||
public AtmosphereProgram(ITamperedProcess process, Parameter<long> pressedKeys, IOperation entryPoint)
|
||||
{
|
||||
Process = process;
|
||||
_pressedKeys = pressedKeys;
|
||||
_entryPoint = entryPoint;
|
||||
}
|
||||
|
||||
public void Execute(ControllerKeys pressedKeys)
|
||||
{
|
||||
_pressedKeys.Value = (long)pressedKeys;
|
||||
_entryPoint.Execute();
|
||||
}
|
||||
}
|
||||
}
|
105
Ryujinx.HLE/HOS/Tamper/CodeEmitters/Arithmetic.cs
Normal file
105
Ryujinx.HLE/HOS/Tamper/CodeEmitters/Arithmetic.cs
Normal file
|
@ -0,0 +1,105 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 9 allows performing arithmetic on registers.
|
||||
/// </summary>
|
||||
class Arithmetic
|
||||
{
|
||||
private const int OperationWidthIndex = 1;
|
||||
private const int OperationTypeIndex = 2;
|
||||
private const int DestinationRegisterIndex = 3;
|
||||
private const int LeftHandSideRegisterIndex = 4;
|
||||
private const int UseImmediateAsRhsIndex = 5;
|
||||
private const int RightHandSideRegisterIndex = 6;
|
||||
private const int RightHandSideImmediateIndex = 8;
|
||||
|
||||
private const int RightHandSideImmediate8 = 8;
|
||||
private const int RightHandSideImmediate16 = 16;
|
||||
|
||||
private const byte Add = 0; // lhs + rhs
|
||||
private const byte Sub = 1; // lhs - rhs
|
||||
private const byte Mul = 2; // lhs * rhs
|
||||
private const byte Lsh = 3; // lhs << rhs
|
||||
private const byte Rsh = 4; // lhs >> rhs
|
||||
private const byte And = 5; // lhs & rhs
|
||||
private const byte Or = 6; // lhs | rhs
|
||||
private const byte Not = 7; // ~lhs (discards right-hand operand)
|
||||
private const byte Xor = 8; // lhs ^ rhs
|
||||
private const byte Mov = 9; // lhs (discards right-hand operand)
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// 9TCRS0s0
|
||||
// T: Width of arithmetic operation(1, 2, 4, or 8 bytes).
|
||||
// C: Arithmetic operation to apply, see below.
|
||||
// R: Register to store result in.
|
||||
// S: Register to use as left - hand operand.
|
||||
// s: Register to use as right - hand operand.
|
||||
|
||||
// 9TCRS100 VVVVVVVV (VVVVVVVV)
|
||||
// T: Width of arithmetic operation(1, 2, 4, or 8 bytes).
|
||||
// C: Arithmetic operation to apply, see below.
|
||||
// R: Register to store result in.
|
||||
// S: Register to use as left - hand operand.
|
||||
// V: Value to use as right - hand operand.
|
||||
|
||||
byte operationWidth = instruction[OperationWidthIndex];
|
||||
byte operation = instruction[OperationTypeIndex];
|
||||
Register destinationRegister = context.GetRegister(instruction[DestinationRegisterIndex]);
|
||||
Register leftHandSideRegister = context.GetRegister(instruction[LeftHandSideRegisterIndex]);
|
||||
byte rightHandSideIsImmediate = instruction[UseImmediateAsRhsIndex];
|
||||
IOperand rightHandSideOperand;
|
||||
|
||||
switch (rightHandSideIsImmediate)
|
||||
{
|
||||
case 0:
|
||||
// Use a register as right-hand side.
|
||||
rightHandSideOperand = context.GetRegister(instruction[RightHandSideRegisterIndex]);
|
||||
break;
|
||||
case 1:
|
||||
// Use an immediate as right-hand side.
|
||||
int immediateSize = operationWidth <= 4 ? RightHandSideImmediate8 : RightHandSideImmediate16;
|
||||
ulong immediate = InstructionHelper.GetImmediate(instruction, RightHandSideImmediateIndex, immediateSize);
|
||||
rightHandSideOperand = new Value<ulong>(immediate);
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid right-hand side switch {rightHandSideIsImmediate} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
void Emit(Type operationType, IOperand rhs = null)
|
||||
{
|
||||
List<IOperand> operandList = new List<IOperand>();
|
||||
operandList.Add(destinationRegister);
|
||||
operandList.Add(leftHandSideRegister);
|
||||
|
||||
if (rhs != null)
|
||||
{
|
||||
operandList.Add(rhs);
|
||||
}
|
||||
|
||||
InstructionHelper.Emit(operationType, operationWidth, context, operandList.ToArray());
|
||||
}
|
||||
|
||||
switch (operation)
|
||||
{
|
||||
case Add: Emit(typeof(OpAdd<>), rightHandSideOperand); break;
|
||||
case Sub: Emit(typeof(OpSub<>), rightHandSideOperand); break;
|
||||
case Mul: Emit(typeof(OpMul<>), rightHandSideOperand); break;
|
||||
case Lsh: Emit(typeof(OpLsh<>), rightHandSideOperand); break;
|
||||
case Rsh: Emit(typeof(OpRsh<>), rightHandSideOperand); break;
|
||||
case And: Emit(typeof(OpAnd<>), rightHandSideOperand); break;
|
||||
case Or: Emit(typeof(OpOr<> ), rightHandSideOperand); break;
|
||||
case Not: Emit(typeof(OpNot<>) ); break;
|
||||
case Xor: Emit(typeof(OpXor<>), rightHandSideOperand); break;
|
||||
case Mov: Emit(typeof(OpMov<>) ); break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid arithmetic operation {operation} in Atmosphere cheat");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
14
Ryujinx.HLE/HOS/Tamper/CodeEmitters/BeginConditionalBlock.cs
Normal file
14
Ryujinx.HLE/HOS/Tamper/CodeEmitters/BeginConditionalBlock.cs
Normal file
|
@ -0,0 +1,14 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Marks the begin of a conditional block (started by Code Type 1, Code Type 8 or Code Type C0).
|
||||
/// </summary>
|
||||
class BeginConditionalBlock
|
||||
{
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// Just start a new compilation block and parse the instruction itself at the end.
|
||||
context.BlockStack.Push(new OperationBlock(instruction));
|
||||
}
|
||||
}
|
||||
}
|
87
Ryujinx.HLE/HOS/Tamper/CodeEmitters/DebugLog.cs
Normal file
87
Ryujinx.HLE/HOS/Tamper/CodeEmitters/DebugLog.cs
Normal file
|
@ -0,0 +1,87 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 0xFFF writes a debug log.
|
||||
/// </summary>
|
||||
class DebugLog
|
||||
{
|
||||
private const int OperationWidthIndex = 3;
|
||||
private const int LogIdIndex = 4;
|
||||
private const int OperandTypeIndex = 5;
|
||||
private const int RegisterOrMemoryRegionIndex = 6;
|
||||
private const int OffsetRegisterOrImmediateIndex = 7;
|
||||
|
||||
private const int MemoryRegionWithOffsetImmediate = 0;
|
||||
private const int MemoryRegionWithOffsetRegister = 1;
|
||||
private const int AddressRegisterWithOffsetImmediate = 2;
|
||||
private const int AddressRegisterWithOffsetRegister = 3;
|
||||
private const int ValueRegister = 4;
|
||||
|
||||
private const int OffsetImmediateSize = 9;
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// FFFTIX##
|
||||
// FFFTI0Ma aaaaaaaa
|
||||
// FFFTI1Mr
|
||||
// FFFTI2Ra aaaaaaaa
|
||||
// FFFTI3Rr
|
||||
// FFFTI4V0
|
||||
// T: Width of memory write (1, 2, 4, or 8 bytes).
|
||||
// I: Log id.
|
||||
// X: Operand Type, see below.
|
||||
// M: Memory Type (operand types 0 and 1).
|
||||
// R: Address Register (operand types 2 and 3).
|
||||
// a: Relative Address (operand types 0 and 2).
|
||||
// r: Offset Register (operand types 1 and 3).
|
||||
// V: Value Register (operand type 4).
|
||||
|
||||
byte operationWidth = instruction[OperationWidthIndex];
|
||||
byte logId = instruction[LogIdIndex];
|
||||
byte operandType = instruction[OperandTypeIndex];
|
||||
byte registerOrMemoryRegion = instruction[RegisterOrMemoryRegionIndex];
|
||||
byte offsetRegisterIndex = instruction[OffsetRegisterOrImmediateIndex];
|
||||
ulong immediate;
|
||||
Register addressRegister;
|
||||
Register offsetRegister;
|
||||
IOperand sourceOperand;
|
||||
|
||||
switch (operandType)
|
||||
{
|
||||
case MemoryRegionWithOffsetImmediate:
|
||||
// *(?x + #a)
|
||||
immediate = InstructionHelper.GetImmediate(instruction, OffsetRegisterOrImmediateIndex, OffsetImmediateSize);
|
||||
sourceOperand = MemoryHelper.EmitPointer((MemoryRegion)registerOrMemoryRegion, immediate, context);
|
||||
break;
|
||||
case MemoryRegionWithOffsetRegister:
|
||||
// *(?x + $r)
|
||||
offsetRegister = context.GetRegister(offsetRegisterIndex);
|
||||
sourceOperand = MemoryHelper.EmitPointer((MemoryRegion)registerOrMemoryRegion, offsetRegister, context);
|
||||
break;
|
||||
case AddressRegisterWithOffsetImmediate:
|
||||
// *($R + #a)
|
||||
addressRegister = context.GetRegister(registerOrMemoryRegion);
|
||||
immediate = InstructionHelper.GetImmediate(instruction, OffsetRegisterOrImmediateIndex, OffsetImmediateSize);
|
||||
sourceOperand = MemoryHelper.EmitPointer(addressRegister, immediate, context);
|
||||
break;
|
||||
case AddressRegisterWithOffsetRegister:
|
||||
// *($R + $r)
|
||||
addressRegister = context.GetRegister(registerOrMemoryRegion);
|
||||
offsetRegister = context.GetRegister(offsetRegisterIndex);
|
||||
sourceOperand = MemoryHelper.EmitPointer(addressRegister, offsetRegister, context);
|
||||
break;
|
||||
case ValueRegister:
|
||||
// $V
|
||||
sourceOperand = context.GetRegister(registerOrMemoryRegion);
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid operand type {operandType} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
InstructionHelper.Emit(typeof(OpLog<>), operationWidth, context, logId, sourceOperand);
|
||||
}
|
||||
}
|
||||
}
|
50
Ryujinx.HLE/HOS/Tamper/CodeEmitters/EndConditionalBlock.cs
Normal file
50
Ryujinx.HLE/HOS/Tamper/CodeEmitters/EndConditionalBlock.cs
Normal file
|
@ -0,0 +1,50 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Conditions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 2 marks the end of a conditional block (started by Code Type 1, Code Type 8 or Code Type C0).
|
||||
/// </summary>
|
||||
class EndConditionalBlock
|
||||
{
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// 20000000
|
||||
|
||||
// Use the conditional begin instruction stored in the stack.
|
||||
instruction = context.CurrentBlock.BaseInstruction;
|
||||
CodeType codeType = InstructionHelper.GetCodeType(instruction);
|
||||
|
||||
// Pop the current block of operations from the stack so control instructions
|
||||
// for the conditional can be emitted in the upper block.
|
||||
IEnumerable<IOperation> operations = context.CurrentOperations;
|
||||
context.BlockStack.Pop();
|
||||
|
||||
ICondition condition;
|
||||
|
||||
switch (codeType)
|
||||
{
|
||||
case CodeType.BeginMemoryConditionalBlock:
|
||||
condition = MemoryConditional.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.BeginKeypressConditionalBlock:
|
||||
condition = KeyPressConditional.Emit(instruction, context);
|
||||
break;
|
||||
case CodeType.BeginRegisterConditionalBlock:
|
||||
condition = RegisterConditional.Emit(instruction, context);
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Conditional end does not match code type {codeType} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
// Create a conditional block with the current operations and nest it in the upper
|
||||
// block of the stack.
|
||||
|
||||
IfBlock block = new IfBlock(condition, operations);
|
||||
context.CurrentOperations.Add(block);
|
||||
}
|
||||
}
|
||||
}
|
26
Ryujinx.HLE/HOS/Tamper/CodeEmitters/KeyPressConditional.cs
Normal file
26
Ryujinx.HLE/HOS/Tamper/CodeEmitters/KeyPressConditional.cs
Normal file
|
@ -0,0 +1,26 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Conditions;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 8 enters or skips a conditional block based on whether a key combination is pressed.
|
||||
/// </summary>
|
||||
class KeyPressConditional
|
||||
{
|
||||
private const int InputMaskIndex = 1;
|
||||
|
||||
private const int InputMaskSize = 7;
|
||||
|
||||
public static ICondition Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// 8kkkkkkk
|
||||
// k: Keypad mask to check against, see below.
|
||||
// Note that for multiple button combinations, the bitmasks should be ORd together.
|
||||
// The Keypad Values are the direct output of hidKeysDown().
|
||||
|
||||
ulong inputMask = InstructionHelper.GetImmediate(instruction, InputMaskIndex, InputMaskSize);
|
||||
|
||||
return new InputMask((long)inputMask, context.PressedKeys);
|
||||
}
|
||||
}
|
||||
}
|
57
Ryujinx.HLE/HOS/Tamper/CodeEmitters/LegacyArithmetic.cs
Normal file
57
Ryujinx.HLE/HOS/Tamper/CodeEmitters/LegacyArithmetic.cs
Normal file
|
@ -0,0 +1,57 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
using System;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 7 allows performing arithmetic on registers. However, it has been deprecated by Code
|
||||
/// type 9, and is only kept for backwards compatibility.
|
||||
/// </summary>
|
||||
class LegacyArithmetic
|
||||
{
|
||||
const int OperationWidthIndex = 1;
|
||||
const int DestinationRegisterIndex = 3;
|
||||
const int OperationTypeIndex = 4;
|
||||
const int ValueImmediateIndex = 8;
|
||||
|
||||
const int ValueImmediateSize = 8;
|
||||
|
||||
private const byte Add = 0; // reg += rhs
|
||||
private const byte Sub = 1; // reg -= rhs
|
||||
private const byte Mul = 2; // reg *= rhs
|
||||
private const byte Lsh = 3; // reg <<= rhs
|
||||
private const byte Rsh = 4; // reg >>= rhs
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// 7T0RC000 VVVVVVVV
|
||||
// T: Width of arithmetic operation(1, 2, 4, or 8 bytes).
|
||||
// R: Register to apply arithmetic to.
|
||||
// C: Arithmetic operation to apply, see below.
|
||||
// V: Value to use for arithmetic operation.
|
||||
|
||||
byte operationWidth = instruction[OperationWidthIndex];
|
||||
Register register = context.GetRegister(instruction[DestinationRegisterIndex]);
|
||||
byte operation = instruction[OperationTypeIndex];
|
||||
ulong immediate = InstructionHelper.GetImmediate(instruction, ValueImmediateIndex, ValueImmediateSize);
|
||||
Value<ulong> rightHandSideValue = new Value<ulong>(immediate);
|
||||
|
||||
void Emit(Type operationType)
|
||||
{
|
||||
InstructionHelper.Emit(operationType, operationWidth, context, register, register, rightHandSideValue);
|
||||
}
|
||||
|
||||
switch (operation)
|
||||
{
|
||||
case Add: Emit(typeof(OpAdd<>)); break;
|
||||
case Sub: Emit(typeof(OpSub<>)); break;
|
||||
case Mul: Emit(typeof(OpMul<>)); break;
|
||||
case Lsh: Emit(typeof(OpLsh<>)); break;
|
||||
case Rsh: Emit(typeof(OpRsh<>)); break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid arithmetic operation {operation} in Atmosphere cheat");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,28 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 4 allows setting a register to a constant value.
|
||||
/// </summary>
|
||||
class LoadRegisterWithConstant
|
||||
{
|
||||
const int RegisterIndex = 3;
|
||||
const int ValueImmediateIndex = 8;
|
||||
|
||||
const int ValueImmediateSize = 16;
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// 400R0000 VVVVVVVV VVVVVVVV
|
||||
// R: Register to use.
|
||||
// V: Value to load.
|
||||
|
||||
Register destinationRegister = context.GetRegister(instruction[RegisterIndex]);
|
||||
ulong immediate = InstructionHelper.GetImmediate(instruction, ValueImmediateIndex, ValueImmediateSize);
|
||||
Value<ulong> sourceValue = new Value<ulong>(immediate);
|
||||
|
||||
context.CurrentOperations.Add(new OpMov<ulong>(destinationRegister, sourceValue));
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,58 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 5 allows loading a value from memory into a register, either using a fixed address or by
|
||||
/// dereferencing the destination register.
|
||||
/// </summary>
|
||||
class LoadRegisterWithMemory
|
||||
{
|
||||
private const int OperationWidthIndex = 1;
|
||||
private const int MemoryRegionIndex = 2;
|
||||
private const int DestinationRegisterIndex = 3;
|
||||
private const int UseDestinationAsSourceIndex = 4;
|
||||
private const int OffsetImmediateIndex = 6;
|
||||
|
||||
private const int OffsetImmediateSize = 10;
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// 5TMR00AA AAAAAAAA
|
||||
// T: Width of memory read (1, 2, 4, or 8 bytes).
|
||||
// M: Memory region to write to (0 = Main NSO, 1 = Heap).
|
||||
// R: Register to load value into.
|
||||
// A: Immediate offset to use from memory region base.
|
||||
|
||||
// 5TMR10AA AAAAAAAA
|
||||
// T: Width of memory read(1, 2, 4, or 8 bytes).
|
||||
// M: Ignored.
|
||||
// R: Register to use as base address and to load value into.
|
||||
// A: Immediate offset to use from register R.
|
||||
|
||||
byte operationWidth = instruction[OperationWidthIndex];
|
||||
MemoryRegion memoryRegion = (MemoryRegion)instruction[MemoryRegionIndex];
|
||||
Register destinationRegister = context.GetRegister(instruction[DestinationRegisterIndex]);
|
||||
byte useDestinationAsSourceIndex = instruction[UseDestinationAsSourceIndex];
|
||||
ulong address = InstructionHelper.GetImmediate(instruction, OffsetImmediateIndex, OffsetImmediateSize);
|
||||
|
||||
Pointer sourceMemory;
|
||||
|
||||
switch (useDestinationAsSourceIndex)
|
||||
{
|
||||
case 0:
|
||||
// Don't use the source register as an additional address offset.
|
||||
sourceMemory = MemoryHelper.EmitPointer(memoryRegion, address, context);
|
||||
break;
|
||||
case 1:
|
||||
// Use the source register as the base address.
|
||||
sourceMemory = MemoryHelper.EmitPointer(destinationRegister, address, context);
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid source mode {useDestinationAsSourceIndex} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
InstructionHelper.EmitMov(operationWidth, context, destinationRegister, sourceMemory);
|
||||
}
|
||||
}
|
||||
}
|
45
Ryujinx.HLE/HOS/Tamper/CodeEmitters/MemoryConditional.cs
Normal file
45
Ryujinx.HLE/HOS/Tamper/CodeEmitters/MemoryConditional.cs
Normal file
|
@ -0,0 +1,45 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Conditions;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 1 performs a comparison of the contents of memory to a static value.
|
||||
/// If the condition is not met, all instructions until the appropriate conditional block terminator
|
||||
/// are skipped.
|
||||
/// </summary>
|
||||
class MemoryConditional
|
||||
{
|
||||
private const int OperationWidthIndex = 1;
|
||||
private const int MemoryRegionIndex = 2;
|
||||
private const int ComparisonTypeIndex = 3;
|
||||
private const int OffsetImmediateIndex = 6;
|
||||
private const int ValueImmediateIndex = 16;
|
||||
|
||||
private const int OffsetImmediateSize = 10;
|
||||
private const int ValueImmediateSize4 = 8;
|
||||
private const int ValueImmediateSize8 = 16;
|
||||
|
||||
public static ICondition Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// 1TMC00AA AAAAAAAA VVVVVVVV (VVVVVVVV)
|
||||
// T: Width of memory write (1, 2, 4, or 8 bytes).
|
||||
// M: Memory region to write to (0 = Main NSO, 1 = Heap).
|
||||
// C: Condition to use, see below.
|
||||
// A: Immediate offset to use from memory region base.
|
||||
// V: Value to compare to.
|
||||
|
||||
byte operationWidth = instruction[OperationWidthIndex];
|
||||
MemoryRegion memoryRegion = (MemoryRegion)instruction[MemoryRegionIndex];
|
||||
Comparison comparison = (Comparison)instruction[ComparisonTypeIndex];
|
||||
|
||||
ulong address = InstructionHelper.GetImmediate(instruction, OffsetImmediateIndex, OffsetImmediateSize);
|
||||
Pointer sourceMemory = MemoryHelper.EmitPointer(memoryRegion, address, context);
|
||||
|
||||
int valueSize = operationWidth <= 4 ? ValueImmediateSize4 : ValueImmediateSize8;
|
||||
ulong value = InstructionHelper.GetImmediate(instruction, ValueImmediateIndex, valueSize);
|
||||
Value<ulong> compareToValue = new Value<ulong>(value);
|
||||
|
||||
return InstructionHelper.CreateCondition(comparison, operationWidth, sourceMemory, compareToValue);
|
||||
}
|
||||
}
|
||||
}
|
17
Ryujinx.HLE/HOS/Tamper/CodeEmitters/PauseProcess.cs
Normal file
17
Ryujinx.HLE/HOS/Tamper/CodeEmitters/PauseProcess.cs
Normal file
|
@ -0,0 +1,17 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 0xFF0 pauses the current process.
|
||||
/// </summary>
|
||||
class PauseProcess
|
||||
{
|
||||
// FF0?????
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
context.CurrentOperations.Add(new OpProcCtrl(context.Process, true));
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,47 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 0xC3 reads or writes a static register with a given register.
|
||||
/// NOTE: Registers are saved and restored to a different set of registers than the ones used
|
||||
/// for the other opcodes (Static Registers).
|
||||
/// </summary>
|
||||
class ReadOrWriteStaticRegister
|
||||
{
|
||||
private const int StaticRegisterIndex = 5;
|
||||
private const int RegisterIndex = 7;
|
||||
|
||||
private const byte FirstWriteRegister = 0x80;
|
||||
|
||||
private const int StaticRegisterSize = 2;
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// C3000XXx
|
||||
// XX: Static register index, 0x00 to 0x7F for reading or 0x80 to 0xFF for writing.
|
||||
// x: Register index.
|
||||
|
||||
ulong staticRegisterIndex = InstructionHelper.GetImmediate(instruction, StaticRegisterIndex, StaticRegisterSize);
|
||||
Register register = context.GetRegister(instruction[RegisterIndex]);
|
||||
|
||||
IOperand sourceRegister;
|
||||
IOperand destinationRegister;
|
||||
|
||||
if (staticRegisterIndex < FirstWriteRegister)
|
||||
{
|
||||
// Read from static register.
|
||||
sourceRegister = context.GetStaticRegister((byte)staticRegisterIndex);
|
||||
destinationRegister = register;
|
||||
}
|
||||
else
|
||||
{
|
||||
// Write to static register.
|
||||
sourceRegister = register;
|
||||
destinationRegister = context.GetStaticRegister((byte)(staticRegisterIndex - FirstWriteRegister));
|
||||
}
|
||||
|
||||
context.CurrentOperations.Add(new OpMov<ulong>(destinationRegister, sourceRegister));
|
||||
}
|
||||
}
|
||||
}
|
106
Ryujinx.HLE/HOS/Tamper/CodeEmitters/RegisterConditional.cs
Normal file
106
Ryujinx.HLE/HOS/Tamper/CodeEmitters/RegisterConditional.cs
Normal file
|
@ -0,0 +1,106 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Conditions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 0xC0 performs a comparison of the contents of a register and another value.
|
||||
/// This code support multiple operand types, see below. If the condition is not met,
|
||||
/// all instructions until the appropriate conditional block terminator are skipped.
|
||||
/// </summary>
|
||||
class RegisterConditional
|
||||
{
|
||||
private const int OperationWidthIndex = 2;
|
||||
private const int ComparisonTypeIndex = 3;
|
||||
private const int SourceRegisterIndex = 4;
|
||||
private const int OperandTypeIndex = 5;
|
||||
private const int RegisterOrMemoryRegionIndex = 6;
|
||||
private const int OffsetImmediateIndex = 7;
|
||||
private const int ValueImmediateIndex = 8;
|
||||
|
||||
private const int MemoryRegionWithOffsetImmediate = 0;
|
||||
private const int MemoryRegionWithOffsetRegister = 1;
|
||||
private const int AddressRegisterWithOffsetImmediate = 2;
|
||||
private const int AddressRegisterWithOffsetRegister = 3;
|
||||
private const int OffsetImmediate = 4;
|
||||
private const int AddressRegister = 5;
|
||||
|
||||
private const int OffsetImmediateSize = 9;
|
||||
private const int ValueImmediateSize8 = 8;
|
||||
private const int ValueImmediateSize16 = 16;
|
||||
|
||||
public static ICondition Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// C0TcSX##
|
||||
// C0TcS0Ma aaaaaaaa
|
||||
// C0TcS1Mr
|
||||
// C0TcS2Ra aaaaaaaa
|
||||
// C0TcS3Rr
|
||||
// C0TcS400 VVVVVVVV (VVVVVVVV)
|
||||
// C0TcS5X0
|
||||
// T: Width of memory write(1, 2, 4, or 8 bytes).
|
||||
// c: Condition to use, see below.
|
||||
// S: Source Register.
|
||||
// X: Operand Type, see below.
|
||||
// M: Memory Type(operand types 0 and 1).
|
||||
// R: Address Register(operand types 2 and 3).
|
||||
// a: Relative Address(operand types 0 and 2).
|
||||
// r: Offset Register(operand types 1 and 3).
|
||||
// X: Other Register(operand type 5).
|
||||
// V: Value to compare to(operand type 4).
|
||||
|
||||
byte operationWidth = instruction[OperationWidthIndex];
|
||||
Comparison comparison = (Comparison)instruction[ComparisonTypeIndex];
|
||||
Register sourceRegister = context.GetRegister(instruction[SourceRegisterIndex]);
|
||||
byte operandType = instruction[OperandTypeIndex];
|
||||
byte registerOrMemoryRegion = instruction[RegisterOrMemoryRegionIndex];
|
||||
byte offsetRegisterIndex = instruction[OffsetImmediateIndex];
|
||||
ulong offsetImmediate;
|
||||
ulong valueImmediate;
|
||||
int valueImmediateSize;
|
||||
Register addressRegister;
|
||||
Register offsetRegister;
|
||||
IOperand sourceOperand;
|
||||
|
||||
switch (operandType)
|
||||
{
|
||||
case MemoryRegionWithOffsetImmediate:
|
||||
// *(?x + #a)
|
||||
offsetImmediate = InstructionHelper.GetImmediate(instruction, OffsetImmediateIndex, OffsetImmediateSize);
|
||||
sourceOperand = MemoryHelper.EmitPointer((MemoryRegion)registerOrMemoryRegion, offsetImmediate, context);
|
||||
break;
|
||||
case MemoryRegionWithOffsetRegister:
|
||||
// *(?x + $r)
|
||||
offsetRegister = context.GetRegister(offsetRegisterIndex);
|
||||
sourceOperand = MemoryHelper.EmitPointer((MemoryRegion)registerOrMemoryRegion, offsetRegister, context);
|
||||
break;
|
||||
case AddressRegisterWithOffsetImmediate:
|
||||
// *($R + #a)
|
||||
addressRegister = context.GetRegister(registerOrMemoryRegion);
|
||||
offsetImmediate = InstructionHelper.GetImmediate(instruction, OffsetImmediateIndex, OffsetImmediateSize);
|
||||
sourceOperand = MemoryHelper.EmitPointer(addressRegister, offsetImmediate, context);
|
||||
break;
|
||||
case AddressRegisterWithOffsetRegister:
|
||||
// *($R + $r)
|
||||
addressRegister = context.GetRegister(registerOrMemoryRegion);
|
||||
offsetRegister = context.GetRegister(offsetRegisterIndex);
|
||||
sourceOperand = MemoryHelper.EmitPointer(addressRegister, offsetRegister, context);
|
||||
break;
|
||||
case OffsetImmediate:
|
||||
valueImmediateSize = operationWidth <= 4 ? ValueImmediateSize8 : ValueImmediateSize16;
|
||||
valueImmediate = InstructionHelper.GetImmediate(instruction, ValueImmediateIndex, valueImmediateSize);
|
||||
sourceOperand = new Value<ulong>(valueImmediate);
|
||||
break;
|
||||
case AddressRegister:
|
||||
// $V
|
||||
sourceOperand = context.GetRegister(registerOrMemoryRegion);
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid operand type {operandType} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
return InstructionHelper.CreateCondition(comparison, operationWidth, sourceRegister, sourceOperand);
|
||||
}
|
||||
}
|
||||
}
|
17
Ryujinx.HLE/HOS/Tamper/CodeEmitters/ResumeProcess.cs
Normal file
17
Ryujinx.HLE/HOS/Tamper/CodeEmitters/ResumeProcess.cs
Normal file
|
@ -0,0 +1,17 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 0xFF1 resumes the current process.
|
||||
/// </summary>
|
||||
class ResumeProcess
|
||||
{
|
||||
// FF1?????
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
context.CurrentOperations.Add(new OpProcCtrl(context.Process, false));
|
||||
}
|
||||
}
|
||||
}
|
65
Ryujinx.HLE/HOS/Tamper/CodeEmitters/SaveOrRestoreRegister.cs
Normal file
65
Ryujinx.HLE/HOS/Tamper/CodeEmitters/SaveOrRestoreRegister.cs
Normal file
|
@ -0,0 +1,65 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 0xC1 performs saving or restoring of registers.
|
||||
/// NOTE: Registers are saved and restored to a different set of registers than the ones used
|
||||
/// for the other opcodes (Save Registers).
|
||||
/// </summary>
|
||||
class SaveOrRestoreRegister
|
||||
{
|
||||
private const int DestinationRegisterIndex = 3;
|
||||
private const int SourceRegisterIndex = 5;
|
||||
private const int OperationTypeIndex = 6;
|
||||
|
||||
private const int RestoreRegister = 0;
|
||||
private const int SaveRegister = 1;
|
||||
private const int ClearSavedValue = 2;
|
||||
private const int ClearRegister = 3;
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// C10D0Sx0
|
||||
// D: Destination index.
|
||||
// S: Source index.
|
||||
// x: Operand Type, see below.
|
||||
|
||||
byte destinationRegIndex = instruction[DestinationRegisterIndex];
|
||||
byte sourceRegIndex = instruction[SourceRegisterIndex];
|
||||
byte operationType = instruction[OperationTypeIndex];
|
||||
Impl(operationType, destinationRegIndex, sourceRegIndex, context);
|
||||
}
|
||||
|
||||
public static void Impl(byte operationType, byte destinationRegIndex, byte sourceRegIndex, CompilationContext context)
|
||||
{
|
||||
IOperand destinationOperand;
|
||||
IOperand sourceOperand;
|
||||
|
||||
switch (operationType)
|
||||
{
|
||||
case RestoreRegister:
|
||||
destinationOperand = context.GetRegister(destinationRegIndex);
|
||||
sourceOperand = context.GetSavedRegister(sourceRegIndex);
|
||||
break;
|
||||
case SaveRegister:
|
||||
destinationOperand = context.GetSavedRegister(destinationRegIndex);
|
||||
sourceOperand = context.GetRegister(sourceRegIndex);
|
||||
break;
|
||||
case ClearSavedValue:
|
||||
destinationOperand = new Value<ulong>(0);
|
||||
sourceOperand = context.GetSavedRegister(sourceRegIndex);
|
||||
break;
|
||||
case ClearRegister:
|
||||
destinationOperand = new Value<ulong>(0);
|
||||
sourceOperand = context.GetRegister(sourceRegIndex);
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid register operation type {operationType} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
context.CurrentOperations.Add(new OpMov<ulong>(destinationOperand, sourceOperand));
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,33 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 0xC2 performs saving or restoring of multiple registers using a bitmask.
|
||||
/// NOTE: Registers are saved and restored to a different set of registers than the ones used
|
||||
/// for the other opcodes (Save Registers).
|
||||
/// </summary>
|
||||
class SaveOrRestoreRegisterWithMask
|
||||
{
|
||||
private const int OperationTypeIndex = 2;
|
||||
private const int RegisterMaskIndex = 4;
|
||||
|
||||
private const int RegisterMaskSize = 4;
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// C2x0XXXX
|
||||
// x: Operand Type, see below.
|
||||
// X: 16-bit bitmask, bit i == save or restore register i.
|
||||
|
||||
byte operationType = instruction[OperationTypeIndex];
|
||||
ulong mask = InstructionHelper.GetImmediate(instruction, RegisterMaskIndex, RegisterMaskSize);
|
||||
|
||||
for (byte regIndex = 0; mask != 0; mask >>= 1, regIndex++)
|
||||
{
|
||||
if ((mask & 0x1) != 0)
|
||||
{
|
||||
SaveOrRestoreRegister.Impl(operationType, regIndex, regIndex, context);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
72
Ryujinx.HLE/HOS/Tamper/CodeEmitters/StartEndLoop.cs
Normal file
72
Ryujinx.HLE/HOS/Tamper/CodeEmitters/StartEndLoop.cs
Normal file
|
@ -0,0 +1,72 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 3 allows for iterating in a loop a fixed number of times.
|
||||
/// </summary>
|
||||
class StartEndLoop
|
||||
{
|
||||
private const int StartOrEndIndex = 1;
|
||||
private const int IterationRegisterIndex = 3;
|
||||
private const int IterationsImmediateIndex = 8;
|
||||
|
||||
private const int IterationsImmediateSize = 8;
|
||||
|
||||
private const byte LoopBegin = 0;
|
||||
private const byte LoopEnd = 1;
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// 300R0000 VVVVVVVV
|
||||
// R: Register to use as loop counter.
|
||||
// V: Number of iterations to loop.
|
||||
|
||||
// 310R0000
|
||||
|
||||
byte mode = instruction[StartOrEndIndex];
|
||||
byte iterationRegisterIndex = instruction[IterationRegisterIndex];
|
||||
|
||||
switch (mode)
|
||||
{
|
||||
case LoopBegin:
|
||||
// Just start a new compilation block and parse the instruction itself at the end.
|
||||
context.BlockStack.Push(new OperationBlock(instruction));
|
||||
return;
|
||||
case LoopEnd:
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid loop {mode} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
// Use the loop begin instruction stored in the stack.
|
||||
instruction = context.CurrentBlock.BaseInstruction;
|
||||
CodeType codeType = InstructionHelper.GetCodeType(instruction);
|
||||
|
||||
if (codeType != CodeType.StartEndLoop)
|
||||
{
|
||||
throw new TamperCompilationException($"Loop end does not match code type {codeType} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
// Validate if the register in the beginning and end are the same.
|
||||
|
||||
byte oldIterationRegisterIndex = instruction[IterationRegisterIndex];
|
||||
|
||||
if (iterationRegisterIndex != oldIterationRegisterIndex)
|
||||
{
|
||||
throw new TamperCompilationException($"The register used for the loop changed from {oldIterationRegisterIndex} to {iterationRegisterIndex} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
Register iterationRegister = context.GetRegister(iterationRegisterIndex);
|
||||
ulong immediate = InstructionHelper.GetImmediate(instruction, IterationsImmediateIndex, IterationsImmediateSize);
|
||||
|
||||
// Create a loop block with the current operations and nest it in the upper
|
||||
// block of the stack.
|
||||
|
||||
ForBlock block = new ForBlock(immediate, iterationRegister, context.CurrentOperations);
|
||||
context.BlockStack.Pop();
|
||||
context.CurrentOperations.Add(block);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,41 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 0 allows writing a static value to a memory address.
|
||||
/// </summary>
|
||||
class StoreConstantToAddress
|
||||
{
|
||||
private const int OperationWidthIndex = 1;
|
||||
private const int MemoryRegionIndex = 2;
|
||||
private const int OffsetRegisterIndex = 3;
|
||||
private const int OffsetImmediateIndex = 6;
|
||||
private const int ValueImmediateIndex = 16;
|
||||
|
||||
private const int OffsetImmediateSize = 10;
|
||||
private const int ValueImmediateSize8 = 8;
|
||||
private const int ValueImmediateSize16 = 16;
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// 0TMR00AA AAAAAAAA VVVVVVVV (VVVVVVVV)
|
||||
// T: Width of memory write(1, 2, 4, or 8 bytes).
|
||||
// M: Memory region to write to(0 = Main NSO, 1 = Heap).
|
||||
// R: Register to use as an offset from memory region base.
|
||||
// A: Immediate offset to use from memory region base.
|
||||
// V: Value to write.
|
||||
|
||||
byte operationWidth = instruction[OperationWidthIndex];
|
||||
MemoryRegion memoryRegion = (MemoryRegion)instruction[MemoryRegionIndex];
|
||||
Register offsetRegister = context.GetRegister(instruction[OffsetRegisterIndex]);
|
||||
ulong offsetImmediate = InstructionHelper.GetImmediate(instruction, OffsetImmediateIndex, OffsetImmediateSize);
|
||||
|
||||
Pointer dstMem = MemoryHelper.EmitPointer(memoryRegion, offsetRegister, offsetImmediate, context);
|
||||
|
||||
int valueImmediateSize = operationWidth <= 4 ? ValueImmediateSize8 : ValueImmediateSize16;
|
||||
ulong valueImmediate = InstructionHelper.GetImmediate(instruction, ValueImmediateIndex, valueImmediateSize);
|
||||
Value<ulong> storeValue = new Value<ulong>(valueImmediate);
|
||||
|
||||
InstructionHelper.EmitMov(operationWidth, context, dstMem, storeValue);
|
||||
}
|
||||
}
|
||||
}
|
71
Ryujinx.HLE/HOS/Tamper/CodeEmitters/StoreConstantToMemory.cs
Normal file
71
Ryujinx.HLE/HOS/Tamper/CodeEmitters/StoreConstantToMemory.cs
Normal file
|
@ -0,0 +1,71 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 6 allows writing a fixed value to a memory address specified by a register.
|
||||
/// </summary>
|
||||
class StoreConstantToMemory
|
||||
{
|
||||
private const int OperationWidthIndex = 1;
|
||||
private const int AddressRegisterIndex = 3;
|
||||
private const int IncrementAddressRegisterIndex = 4;
|
||||
private const int UseOffsetRegisterIndex = 5;
|
||||
private const int OffsetRegisterIndex = 6;
|
||||
private const int ValueImmediateIndex = 8;
|
||||
|
||||
private const int ValueImmediateSize = 16;
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// 6T0RIor0 VVVVVVVV VVVVVVVV
|
||||
// T: Width of memory write(1, 2, 4, or 8 bytes).
|
||||
// R: Register used as base memory address.
|
||||
// I: Increment register flag(0 = do not increment R, 1 = increment R by T).
|
||||
// o: Offset register enable flag(0 = do not add r to address, 1 = add r to address).
|
||||
// r: Register used as offset when o is 1.
|
||||
// V: Value to write to memory.
|
||||
|
||||
byte operationWidth = instruction[OperationWidthIndex];
|
||||
Register sourceRegister = context.GetRegister(instruction[AddressRegisterIndex]);
|
||||
byte incrementAddressRegister = instruction[IncrementAddressRegisterIndex];
|
||||
byte useOffsetRegister = instruction[UseOffsetRegisterIndex];
|
||||
ulong immediate = InstructionHelper.GetImmediate(instruction, ValueImmediateIndex, ValueImmediateSize);
|
||||
Value<ulong> storeValue = new Value<ulong>(immediate);
|
||||
|
||||
Pointer destinationMemory;
|
||||
|
||||
switch (useOffsetRegister)
|
||||
{
|
||||
case 0:
|
||||
// Don't offset the address register by another register.
|
||||
destinationMemory = MemoryHelper.EmitPointer(sourceRegister, context);
|
||||
break;
|
||||
case 1:
|
||||
// Replace the source address by the sum of the base and offset registers.
|
||||
Register offsetRegister = context.GetRegister(instruction[OffsetRegisterIndex]);
|
||||
destinationMemory = MemoryHelper.EmitPointer(sourceRegister, offsetRegister, context);
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid offset mode {useOffsetRegister} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
InstructionHelper.EmitMov(operationWidth, context, destinationMemory, storeValue);
|
||||
|
||||
switch (incrementAddressRegister)
|
||||
{
|
||||
case 0:
|
||||
// Don't increment the address register by operationWidth.
|
||||
break;
|
||||
case 1:
|
||||
// Increment the address register by operationWidth.
|
||||
IOperand increment = new Value<ulong>(operationWidth);
|
||||
context.CurrentOperations.Add(new OpAdd<ulong>(sourceRegister, sourceRegister, increment));
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid increment mode {incrementAddressRegister} in Atmosphere cheat");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
99
Ryujinx.HLE/HOS/Tamper/CodeEmitters/StoreRegisterToMemory.cs
Normal file
99
Ryujinx.HLE/HOS/Tamper/CodeEmitters/StoreRegisterToMemory.cs
Normal file
|
@ -0,0 +1,99 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 10 allows writing a register to memory.
|
||||
/// </summary>
|
||||
class StoreRegisterToMemory
|
||||
{
|
||||
private const int OperationWidthIndex = 1;
|
||||
private const int SourceRegisterIndex = 2;
|
||||
private const int AddressRegisterIndex = 3;
|
||||
private const int IncrementAddressRegisterIndex = 4;
|
||||
private const int AddressingTypeIndex = 5;
|
||||
private const int RegisterOrMemoryRegionIndex = 6;
|
||||
private const int OffsetImmediateIndex = 7;
|
||||
|
||||
private const int AddressRegister = 0;
|
||||
private const int AddressRegisterWithOffsetRegister = 1;
|
||||
private const int OffsetImmediate = 2;
|
||||
private const int MemoryRegionWithOffsetRegister = 3;
|
||||
private const int MemoryRegionWithOffsetImmediate = 4;
|
||||
private const int MemoryRegionWithOffsetRegisterAndImmediate = 5;
|
||||
|
||||
private const int OffsetImmediateSize1 = 1;
|
||||
private const int OffsetImmediateSize9 = 9;
|
||||
|
||||
public static void Emit(byte[] instruction, CompilationContext context)
|
||||
{
|
||||
// ATSRIOxa (aaaaaaaa)
|
||||
// T: Width of memory write (1, 2, 4, or 8 bytes).
|
||||
// S: Register to write to memory.
|
||||
// R: Register to use as base address.
|
||||
// I: Increment register flag (0 = do not increment R, 1 = increment R by T).
|
||||
// O: Offset type, see below.
|
||||
// x: Register used as offset when O is 1, Memory type when O is 3, 4 or 5.
|
||||
// a: Value used as offset when O is 2, 4 or 5.
|
||||
|
||||
byte operationWidth = instruction[OperationWidthIndex];
|
||||
Register sourceRegister = context.GetRegister(instruction[SourceRegisterIndex]);
|
||||
Register addressRegister = context.GetRegister(instruction[AddressRegisterIndex]);
|
||||
byte incrementAddressRegister = instruction[IncrementAddressRegisterIndex];
|
||||
byte offsetType = instruction[AddressingTypeIndex];
|
||||
byte registerOrMemoryRegion = instruction[RegisterOrMemoryRegionIndex];
|
||||
int immediateSize = instruction.Length <= 8 ? OffsetImmediateSize1 : OffsetImmediateSize9;
|
||||
ulong immediate = InstructionHelper.GetImmediate(instruction, OffsetImmediateIndex, immediateSize);
|
||||
|
||||
Pointer destinationMemory;
|
||||
|
||||
switch (offsetType)
|
||||
{
|
||||
case AddressRegister:
|
||||
// *($R) = $S
|
||||
destinationMemory = MemoryHelper.EmitPointer(addressRegister, context);
|
||||
break;
|
||||
case AddressRegisterWithOffsetRegister:
|
||||
// *($R + $x) = $S
|
||||
Register offsetRegister = context.GetRegister(registerOrMemoryRegion);
|
||||
destinationMemory = MemoryHelper.EmitPointer(addressRegister, offsetRegister, context);
|
||||
break;
|
||||
case OffsetImmediate:
|
||||
// *(#a) = $S
|
||||
destinationMemory = MemoryHelper.EmitPointer(addressRegister, immediate, context);
|
||||
break;
|
||||
case MemoryRegionWithOffsetRegister:
|
||||
// *(?x + $R) = $S
|
||||
destinationMemory = MemoryHelper.EmitPointer((MemoryRegion)registerOrMemoryRegion, addressRegister, context);
|
||||
break;
|
||||
case MemoryRegionWithOffsetImmediate:
|
||||
// *(?x + #a) = $S
|
||||
destinationMemory = MemoryHelper.EmitPointer((MemoryRegion)registerOrMemoryRegion, immediate, context);
|
||||
break;
|
||||
case MemoryRegionWithOffsetRegisterAndImmediate:
|
||||
// *(?x + #a + $R) = $S
|
||||
destinationMemory = MemoryHelper.EmitPointer((MemoryRegion)registerOrMemoryRegion, addressRegister, immediate, context);
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid offset type {offsetType} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
InstructionHelper.EmitMov(operationWidth, context, destinationMemory, sourceRegister);
|
||||
|
||||
switch (incrementAddressRegister)
|
||||
{
|
||||
case 0:
|
||||
// Don't increment the address register by operationWidth.
|
||||
break;
|
||||
case 1:
|
||||
// Increment the address register by operationWidth.
|
||||
IOperand increment = new Value<ulong>(operationWidth);
|
||||
context.CurrentOperations.Add(new OpAdd<ulong>(addressRegister, addressRegister, increment));
|
||||
break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid increment mode {incrementAddressRegister} in Atmosphere cheat");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
116
Ryujinx.HLE/HOS/Tamper/CodeType.cs
Normal file
116
Ryujinx.HLE/HOS/Tamper/CodeType.cs
Normal file
|
@ -0,0 +1,116 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
/// <summary>
|
||||
/// The opcodes specified for the Atmosphere Cheat VM.
|
||||
/// </summary>
|
||||
enum CodeType
|
||||
{
|
||||
/// <summary>
|
||||
/// Code type 0 allows writing a static value to a memory address.
|
||||
/// </summary>
|
||||
StoreConstantToAddress = 0x0,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 1 performs a comparison of the contents of memory to a static value.
|
||||
/// If the condition is not met, all instructions until the appropriate conditional block terminator
|
||||
/// are skipped.
|
||||
/// </summary>
|
||||
BeginMemoryConditionalBlock = 0x1,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 2 marks the end of a conditional block (started by Code Type 1 or Code Type 8).
|
||||
/// </summary>
|
||||
EndConditionalBlock = 0x2,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 3 allows for iterating in a loop a fixed number of times.
|
||||
/// </summary>
|
||||
StartEndLoop = 0x3,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 4 allows setting a register to a constant value.
|
||||
/// </summary>
|
||||
LoadRegisterWithContant = 0x4,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 5 allows loading a value from memory into a register, either using a fixed address or by
|
||||
/// dereferencing the destination register.
|
||||
/// </summary>
|
||||
LoadRegisterWithMemory = 0x5,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 6 allows writing a fixed value to a memory address specified by a register.
|
||||
/// </summary>
|
||||
StoreConstantToMemory = 0x6,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 7 allows performing arithmetic on registers. However, it has been deprecated by Code
|
||||
/// type 9, and is only kept for backwards compatibility.
|
||||
/// </summary>
|
||||
LegacyArithmetic = 0x7,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 8 enters or skips a conditional block based on whether a key combination is pressed.
|
||||
/// </summary>
|
||||
BeginKeypressConditionalBlock = 0x8,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 9 allows performing arithmetic on registers.
|
||||
/// </summary>
|
||||
Arithmetic = 0x9,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 10 allows writing a register to memory.
|
||||
/// </summary>
|
||||
StoreRegisterToMemory = 0xA,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 0xC0 performs a comparison of the contents of a register and another value.
|
||||
/// This code support multiple operand types, see below. If the condition is not met,
|
||||
/// all instructions until the appropriate conditional block terminator are skipped.
|
||||
/// </summary>
|
||||
BeginRegisterConditionalBlock = 0xC0,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 0xC1 performs saving or restoring of registers.
|
||||
/// NOTE: Registers are saved and restored to a different set of registers than the ones used
|
||||
/// for the other opcodes (Save Registers).
|
||||
/// </summary>
|
||||
SaveOrRestoreRegister = 0xC1,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 0xC2 performs saving or restoring of multiple registers using a bitmask.
|
||||
/// NOTE: Registers are saved and restored to a different set of registers than the ones used
|
||||
/// for the other opcodes (Save Registers).
|
||||
/// </summary>
|
||||
SaveOrRestoreRegisterWithMask = 0xC2,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 0xC3 reads or writes a static register with a given register.
|
||||
/// NOTE: Registers are saved and restored to a different set of registers than the ones used
|
||||
/// for the other opcodes (Static Registers).
|
||||
/// </summary>
|
||||
ReadOrWriteStaticRegister = 0xC3,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 0xFF0 pauses the current process.
|
||||
/// </summary>
|
||||
PauseProcess = 0xFF0,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 0xFF1 resumes the current process.
|
||||
/// </summary>
|
||||
ResumeProcess = 0xFF1,
|
||||
|
||||
/// <summary>
|
||||
/// Code type 0xFFF writes a debug log.
|
||||
/// </summary>
|
||||
DebugLog = 0xFFF
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Comparison.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Comparison.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
/// <summary>
|
||||
/// The comparisons used by conditional operations.
|
||||
/// </summary>
|
||||
enum Comparison
|
||||
{
|
||||
Greater = 1,
|
||||
GreaterOrEqual = 2,
|
||||
Less = 3,
|
||||
LessOrEqual = 4,
|
||||
Equal = 5,
|
||||
NotEqual = 6
|
||||
}
|
||||
}
|
75
Ryujinx.HLE/HOS/Tamper/CompilationContext.cs
Normal file
75
Ryujinx.HLE/HOS/Tamper/CompilationContext.cs
Normal file
|
@ -0,0 +1,75 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
class CompilationContext
|
||||
{
|
||||
public OperationBlock CurrentBlock => BlockStack.Peek();
|
||||
public List<IOperation> CurrentOperations => CurrentBlock.Operations;
|
||||
|
||||
public ITamperedProcess Process { get; }
|
||||
public Parameter<long> PressedKeys { get; }
|
||||
public Stack<OperationBlock> BlockStack { get; }
|
||||
public Dictionary<byte, Register> Registers { get; }
|
||||
public Dictionary<byte, Register> SavedRegisters { get; }
|
||||
public Dictionary<byte, Register> StaticRegisters { get; }
|
||||
public ulong ExeAddress { get; }
|
||||
public ulong HeapAddress { get; }
|
||||
|
||||
public CompilationContext(ulong exeAddress, ulong heapAddress, ITamperedProcess process)
|
||||
{
|
||||
Process = process;
|
||||
PressedKeys = new Parameter<long>(0);
|
||||
BlockStack = new Stack<OperationBlock>();
|
||||
Registers = new Dictionary<byte, Register>();
|
||||
SavedRegisters = new Dictionary<byte, Register>();
|
||||
StaticRegisters = new Dictionary<byte, Register>();
|
||||
ExeAddress = exeAddress;
|
||||
HeapAddress = heapAddress;
|
||||
}
|
||||
|
||||
public Register GetRegister(byte index)
|
||||
{
|
||||
if (Registers.TryGetValue(index, out Register register))
|
||||
{
|
||||
return register;
|
||||
}
|
||||
|
||||
register = new Register($"R_{index:X2}");
|
||||
Registers.Add(index, register);
|
||||
|
||||
return register;
|
||||
}
|
||||
|
||||
public Register GetSavedRegister(byte index)
|
||||
{
|
||||
if (SavedRegisters.TryGetValue(index, out Register register))
|
||||
{
|
||||
return register;
|
||||
}
|
||||
|
||||
register = new Register($"S_{index:X2}");
|
||||
SavedRegisters.Add(index, register);
|
||||
|
||||
return register;
|
||||
}
|
||||
|
||||
public Register GetStaticRegister(byte index)
|
||||
{
|
||||
if (SavedRegisters.TryGetValue(index, out Register register))
|
||||
{
|
||||
return register;
|
||||
}
|
||||
|
||||
register = new Register($"T_{index:X2}");
|
||||
SavedRegisters.Add(index, register);
|
||||
|
||||
return register;
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondEQ.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondEQ.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.Conditions
|
||||
{
|
||||
class CondEQ<T> : ICondition where T : unmanaged
|
||||
{
|
||||
private IOperand _lhs;
|
||||
private IOperand _rhs;
|
||||
|
||||
public CondEQ(IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public bool Evaluate()
|
||||
{
|
||||
return (dynamic)_lhs.Get<T>() == (dynamic)_rhs.Get<T>();
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondGE.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondGE.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.Conditions
|
||||
{
|
||||
class CondGE<T> : ICondition where T : unmanaged
|
||||
{
|
||||
private IOperand _lhs;
|
||||
private IOperand _rhs;
|
||||
|
||||
public CondGE(IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public bool Evaluate()
|
||||
{
|
||||
return (dynamic)_lhs.Get<T>() >= (dynamic)_rhs.Get<T>();
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondGT.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondGT.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.Conditions
|
||||
{
|
||||
class CondGT<T> : ICondition where T : unmanaged
|
||||
{
|
||||
private IOperand _lhs;
|
||||
private IOperand _rhs;
|
||||
|
||||
public CondGT(IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public bool Evaluate()
|
||||
{
|
||||
return (dynamic)_lhs.Get<T>() > (dynamic)_rhs.Get<T>();
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondLE.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondLE.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.Conditions
|
||||
{
|
||||
class CondLE<T> : ICondition where T : unmanaged
|
||||
{
|
||||
private IOperand _lhs;
|
||||
private IOperand _rhs;
|
||||
|
||||
public CondLE(IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public bool Evaluate()
|
||||
{
|
||||
return (dynamic)_lhs.Get<T>() <= (dynamic)_rhs.Get<T>();
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondLT.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondLT.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.Conditions
|
||||
{
|
||||
class CondLT<T> : ICondition where T : unmanaged
|
||||
{
|
||||
private IOperand _lhs;
|
||||
private IOperand _rhs;
|
||||
|
||||
public CondLT(IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public bool Evaluate()
|
||||
{
|
||||
return (dynamic)_lhs.Get<T>() < (dynamic)_rhs.Get<T>();
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondNE.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Conditions/CondNE.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.Conditions
|
||||
{
|
||||
class CondNE<T> : ICondition where T : unmanaged
|
||||
{
|
||||
private IOperand _lhs;
|
||||
private IOperand _rhs;
|
||||
|
||||
public CondNE(IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public bool Evaluate()
|
||||
{
|
||||
return (dynamic)_lhs.Get<T>() != (dynamic)_rhs.Get<T>();
|
||||
}
|
||||
}
|
||||
}
|
7
Ryujinx.HLE/HOS/Tamper/Conditions/ICondition.cs
Normal file
7
Ryujinx.HLE/HOS/Tamper/Conditions/ICondition.cs
Normal file
|
@ -0,0 +1,7 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Conditions
|
||||
{
|
||||
interface ICondition
|
||||
{
|
||||
bool Evaluate();
|
||||
}
|
||||
}
|
19
Ryujinx.HLE/HOS/Tamper/Conditions/InputMask.cs
Normal file
19
Ryujinx.HLE/HOS/Tamper/Conditions/InputMask.cs
Normal file
|
@ -0,0 +1,19 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Conditions
|
||||
{
|
||||
class InputMask : ICondition
|
||||
{
|
||||
private long _mask;
|
||||
private Parameter<long> _input;
|
||||
|
||||
public InputMask(long mask, Parameter<long> input)
|
||||
{
|
||||
_mask = mask;
|
||||
_input = input;
|
||||
}
|
||||
|
||||
public bool Evaluate()
|
||||
{
|
||||
return (_input.Value & _mask) != 0;
|
||||
}
|
||||
}
|
||||
}
|
10
Ryujinx.HLE/HOS/Tamper/ITamperProgram.cs
Normal file
10
Ryujinx.HLE/HOS/Tamper/ITamperProgram.cs
Normal file
|
@ -0,0 +1,10 @@
|
|||
using Ryujinx.HLE.HOS.Services.Hid;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
interface ITamperProgram
|
||||
{
|
||||
ITamperedProcess Process { get; }
|
||||
void Execute(ControllerKeys pressedKeys);
|
||||
}
|
||||
}
|
13
Ryujinx.HLE/HOS/Tamper/ITamperedProcess.cs
Normal file
13
Ryujinx.HLE/HOS/Tamper/ITamperedProcess.cs
Normal file
|
@ -0,0 +1,13 @@
|
|||
using Ryujinx.HLE.HOS.Kernel.Process;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
interface ITamperedProcess
|
||||
{
|
||||
ProcessState State { get; }
|
||||
T ReadMemory<T>(ulong va) where T : unmanaged;
|
||||
void WriteMemory<T>(ulong va, T value) where T : unmanaged;
|
||||
void PauseProcess();
|
||||
void ResumeProcess();
|
||||
}
|
||||
}
|
134
Ryujinx.HLE/HOS/Tamper/InstructionHelper.cs
Normal file
134
Ryujinx.HLE/HOS/Tamper/InstructionHelper.cs
Normal file
|
@ -0,0 +1,134 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Conditions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
using System;
|
||||
using System.Globalization;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
class InstructionHelper
|
||||
{
|
||||
private const int CodeTypeIndex = 0;
|
||||
|
||||
public static void Emit(IOperation operation, CompilationContext context)
|
||||
{
|
||||
context.CurrentOperations.Add(operation);
|
||||
}
|
||||
|
||||
public static void Emit(Type instruction, byte width, CompilationContext context, params Object[] operands)
|
||||
{
|
||||
Emit((IOperation)Create(instruction, width, operands), context);
|
||||
}
|
||||
|
||||
public static void EmitMov(byte width, CompilationContext context, IOperand destination, IOperand source)
|
||||
{
|
||||
Emit(typeof(OpMov<>), width, context, destination, source);
|
||||
}
|
||||
|
||||
public static ICondition CreateCondition(Comparison comparison, byte width, IOperand lhs, IOperand rhs)
|
||||
{
|
||||
ICondition Create(Type conditionType)
|
||||
{
|
||||
return (ICondition)InstructionHelper.Create(conditionType, width, lhs, rhs);
|
||||
}
|
||||
|
||||
switch (comparison)
|
||||
{
|
||||
case Comparison.Greater : return Create(typeof(CondGT<>));
|
||||
case Comparison.GreaterOrEqual: return Create(typeof(CondGE<>));
|
||||
case Comparison.Less : return Create(typeof(CondLT<>));
|
||||
case Comparison.LessOrEqual : return Create(typeof(CondLE<>));
|
||||
case Comparison.Equal : return Create(typeof(CondEQ<>));
|
||||
case Comparison.NotEqual : return Create(typeof(CondNE<>));
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid comparison {comparison} in Atmosphere cheat");
|
||||
}
|
||||
}
|
||||
|
||||
public static Object Create(Type instruction, byte width, params Object[] operands)
|
||||
{
|
||||
Type realType;
|
||||
|
||||
switch (width)
|
||||
{
|
||||
case 1: realType = instruction.MakeGenericType(typeof(byte)); break;
|
||||
case 2: realType = instruction.MakeGenericType(typeof(ushort)); break;
|
||||
case 4: realType = instruction.MakeGenericType(typeof(uint)); break;
|
||||
case 8: realType = instruction.MakeGenericType(typeof(ulong)); break;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid instruction width {width} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
return Activator.CreateInstance(realType, operands);
|
||||
}
|
||||
|
||||
public static ulong GetImmediate(byte[] instruction, int index, int nybbleCount)
|
||||
{
|
||||
ulong value = 0;
|
||||
|
||||
for (int i = 0; i < nybbleCount; i++)
|
||||
{
|
||||
value <<= 4;
|
||||
value |= instruction[index + i];
|
||||
}
|
||||
|
||||
return value;
|
||||
}
|
||||
|
||||
public static CodeType GetCodeType(byte[] instruction)
|
||||
{
|
||||
int codeType = instruction[CodeTypeIndex];
|
||||
|
||||
if (codeType >= 0xC)
|
||||
{
|
||||
byte extension = instruction[CodeTypeIndex + 1];
|
||||
codeType = (codeType << 4) | extension;
|
||||
|
||||
if (extension == 0xF)
|
||||
{
|
||||
extension = instruction[CodeTypeIndex + 2];
|
||||
codeType = (codeType << 4) | extension;
|
||||
}
|
||||
}
|
||||
|
||||
return (CodeType)codeType;
|
||||
}
|
||||
|
||||
public static byte[] ParseRawInstruction(string rawInstruction)
|
||||
{
|
||||
const int wordSize = 2 * sizeof(uint);
|
||||
|
||||
// Instructions are multi-word, with 32bit words. Split the raw instruction
|
||||
// and parse each word into individual nybbles of bits.
|
||||
|
||||
var words = rawInstruction.Split((char[])null, StringSplitOptions.RemoveEmptyEntries);
|
||||
|
||||
byte[] instruction = new byte[wordSize * words.Length];
|
||||
|
||||
if (words.Length == 0)
|
||||
{
|
||||
throw new TamperCompilationException("Empty instruction in Atmosphere cheat");
|
||||
}
|
||||
|
||||
for (int wordIndex = 0; wordIndex < words.Length; wordIndex++)
|
||||
{
|
||||
string word = words[wordIndex];
|
||||
|
||||
if (word.Length != wordSize)
|
||||
{
|
||||
throw new TamperCompilationException($"Invalid word length for {word} in Atmosphere cheat");
|
||||
}
|
||||
|
||||
for (int nybbleIndex = 0; nybbleIndex < wordSize; nybbleIndex++)
|
||||
{
|
||||
int index = wordIndex * wordSize + nybbleIndex;
|
||||
string byteData = word.Substring(nybbleIndex, 1);
|
||||
|
||||
instruction[index] = byte.Parse(byteData, NumberStyles.HexNumber, CultureInfo.InvariantCulture);
|
||||
}
|
||||
}
|
||||
|
||||
return instruction;
|
||||
}
|
||||
}
|
||||
}
|
89
Ryujinx.HLE/HOS/Tamper/MemoryHelper.cs
Normal file
89
Ryujinx.HLE/HOS/Tamper/MemoryHelper.cs
Normal file
|
@ -0,0 +1,89 @@
|
|||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
class MemoryHelper
|
||||
{
|
||||
public static ulong GetAddressShift(MemoryRegion source, CompilationContext context)
|
||||
{
|
||||
switch (source)
|
||||
{
|
||||
case MemoryRegion.NSO:
|
||||
// Memory address is relative to the code start.
|
||||
return context.ExeAddress;
|
||||
case MemoryRegion.Heap:
|
||||
// Memory address is relative to the heap.
|
||||
return context.HeapAddress;
|
||||
default:
|
||||
throw new TamperCompilationException($"Invalid memory source {source} in Atmosphere cheat");
|
||||
}
|
||||
}
|
||||
|
||||
private static void EmitAdd(Value<ulong> finalValue, IOperand firstOperand, IOperand secondOperand, CompilationContext context)
|
||||
{
|
||||
context.CurrentOperations.Add(new OpAdd<ulong>(finalValue, firstOperand, secondOperand));
|
||||
}
|
||||
|
||||
public static Pointer EmitPointer(ulong addressImmediate, CompilationContext context)
|
||||
{
|
||||
Value<ulong> addressImmediateValue = new Value<ulong>(addressImmediate);
|
||||
|
||||
return new Pointer(addressImmediateValue, context.Process);
|
||||
}
|
||||
|
||||
public static Pointer EmitPointer(Register addressRegister, CompilationContext context)
|
||||
{
|
||||
return new Pointer(addressRegister, context.Process);
|
||||
}
|
||||
|
||||
public static Pointer EmitPointer(Register addressRegister, ulong offsetImmediate, CompilationContext context)
|
||||
{
|
||||
Value<ulong> offsetImmediateValue = new Value<ulong>(offsetImmediate);
|
||||
Value<ulong> finalAddressValue = new Value<ulong>(0);
|
||||
EmitAdd(finalAddressValue, addressRegister, offsetImmediateValue, context);
|
||||
|
||||
return new Pointer(finalAddressValue, context.Process);
|
||||
}
|
||||
|
||||
public static Pointer EmitPointer(Register addressRegister, Register offsetRegister, CompilationContext context)
|
||||
{
|
||||
Value<ulong> finalAddressValue = new Value<ulong>(0);
|
||||
EmitAdd(finalAddressValue, addressRegister, offsetRegister, context);
|
||||
|
||||
return new Pointer(finalAddressValue, context.Process);
|
||||
}
|
||||
|
||||
public static Pointer EmitPointer(Register addressRegister, Register offsetRegister, ulong offsetImmediate, CompilationContext context)
|
||||
{
|
||||
Value<ulong> offsetImmediateValue = new Value<ulong>(offsetImmediate);
|
||||
Value<ulong> finalOffsetValue = new Value<ulong>(0);
|
||||
EmitAdd(finalOffsetValue, offsetRegister, offsetImmediateValue, context);
|
||||
Value<ulong> finalAddressValue = new Value<ulong>(0);
|
||||
EmitAdd(finalAddressValue, addressRegister, finalOffsetValue, context);
|
||||
|
||||
return new Pointer(finalAddressValue, context.Process);
|
||||
}
|
||||
|
||||
public static Pointer EmitPointer(MemoryRegion memoryRegion, ulong offsetImmediate, CompilationContext context)
|
||||
{
|
||||
offsetImmediate += GetAddressShift(memoryRegion, context);
|
||||
|
||||
return EmitPointer(offsetImmediate, context);
|
||||
}
|
||||
|
||||
public static Pointer EmitPointer(MemoryRegion memoryRegion, Register offsetRegister, CompilationContext context)
|
||||
{
|
||||
ulong offsetImmediate = GetAddressShift(memoryRegion, context);
|
||||
|
||||
return EmitPointer(offsetRegister, offsetImmediate, context);
|
||||
}
|
||||
|
||||
public static Pointer EmitPointer(MemoryRegion memoryRegion, Register offsetRegister, ulong offsetImmediate, CompilationContext context)
|
||||
{
|
||||
offsetImmediate += GetAddressShift(memoryRegion, context);
|
||||
|
||||
return EmitPointer(offsetRegister, offsetImmediate, context);
|
||||
}
|
||||
}
|
||||
}
|
25
Ryujinx.HLE/HOS/Tamper/MemoryRegion.cs
Normal file
25
Ryujinx.HLE/HOS/Tamper/MemoryRegion.cs
Normal file
|
@ -0,0 +1,25 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
/// <summary>
|
||||
/// The regions in the virtual address space of the process that are used as base address of memory operations.
|
||||
/// </summary>
|
||||
enum MemoryRegion
|
||||
{
|
||||
/// <summary>
|
||||
/// The position of the NSO associated with the cheat in the virtual address space.
|
||||
/// NOTE: A game can have several NSOs, but the cheat only associates itself with one.
|
||||
/// </summary>
|
||||
NSO = 0x0,
|
||||
|
||||
/// <summary>
|
||||
/// The address of the heap, as determined by the kernel.
|
||||
/// </summary>
|
||||
Heap = 0x1
|
||||
}
|
||||
}
|
17
Ryujinx.HLE/HOS/Tamper/OperationBlock.cs
Normal file
17
Ryujinx.HLE/HOS/Tamper/OperationBlock.cs
Normal file
|
@ -0,0 +1,17 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
struct OperationBlock
|
||||
{
|
||||
public byte[] BaseInstruction { get; }
|
||||
public List<IOperation> Operations { get; }
|
||||
|
||||
public OperationBlock(byte[] baseInstruction)
|
||||
{
|
||||
BaseInstruction = baseInstruction;
|
||||
Operations = new List<IOperation>();
|
||||
}
|
||||
}
|
||||
}
|
27
Ryujinx.HLE/HOS/Tamper/Operations/Block.cs
Normal file
27
Ryujinx.HLE/HOS/Tamper/Operations/Block.cs
Normal file
|
@ -0,0 +1,27 @@
|
|||
using System.Collections.Generic;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class Block : IOperation
|
||||
{
|
||||
private IEnumerable<IOperation> _operations;
|
||||
|
||||
public Block(IEnumerable<IOperation> operations)
|
||||
{
|
||||
_operations = operations;
|
||||
}
|
||||
|
||||
public Block(params IOperation[] operations)
|
||||
{
|
||||
_operations = operations;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
foreach (IOperation op in _operations)
|
||||
{
|
||||
op.Execute();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
42
Ryujinx.HLE/HOS/Tamper/Operations/ForBlock.cs
Normal file
42
Ryujinx.HLE/HOS/Tamper/Operations/ForBlock.cs
Normal file
|
@ -0,0 +1,42 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Conditions;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class ForBlock : IOperation
|
||||
{
|
||||
private ulong _count;
|
||||
private Register _register;
|
||||
private IEnumerable<IOperation> _operations;
|
||||
|
||||
public ForBlock(ulong count, Register register, IEnumerable<IOperation> operations)
|
||||
{
|
||||
_count = count;
|
||||
_register = register;
|
||||
_operations = operations;
|
||||
}
|
||||
|
||||
public ForBlock(ulong count, Register register, params IOperation[] operations)
|
||||
{
|
||||
_count = count;
|
||||
_register = register;
|
||||
_operations = operations;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
for (ulong i = 0; i < _count; i++)
|
||||
{
|
||||
// Set the register and execute the operations so that changing the
|
||||
// register during runtime does not break iteration.
|
||||
|
||||
_register.Set<ulong>(i);
|
||||
|
||||
foreach (IOperation op in _operations)
|
||||
{
|
||||
op.Execute();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
8
Ryujinx.HLE/HOS/Tamper/Operations/IOperand.cs
Normal file
8
Ryujinx.HLE/HOS/Tamper/Operations/IOperand.cs
Normal file
|
@ -0,0 +1,8 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
interface IOperand
|
||||
{
|
||||
public T Get<T>() where T : unmanaged;
|
||||
public void Set<T>(T value) where T : unmanaged;
|
||||
}
|
||||
}
|
7
Ryujinx.HLE/HOS/Tamper/Operations/IOperation.cs
Normal file
7
Ryujinx.HLE/HOS/Tamper/Operations/IOperation.cs
Normal file
|
@ -0,0 +1,7 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
interface IOperation
|
||||
{
|
||||
void Execute();
|
||||
}
|
||||
}
|
35
Ryujinx.HLE/HOS/Tamper/Operations/IfBlock.cs
Normal file
35
Ryujinx.HLE/HOS/Tamper/Operations/IfBlock.cs
Normal file
|
@ -0,0 +1,35 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Conditions;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class IfBlock : IOperation
|
||||
{
|
||||
private ICondition _condition;
|
||||
private IEnumerable<IOperation> _operations;
|
||||
|
||||
public IfBlock(ICondition condition, IEnumerable<IOperation> operations)
|
||||
{
|
||||
_condition = condition;
|
||||
_operations = operations;
|
||||
}
|
||||
|
||||
public IfBlock(ICondition condition, params IOperation[] operations)
|
||||
{
|
||||
_operations = operations;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
if (!_condition.Evaluate())
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
foreach (IOperation op in _operations)
|
||||
{
|
||||
op.Execute();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Operations/OpAdd.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Operations/OpAdd.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpAdd<T> : IOperation where T : unmanaged
|
||||
{
|
||||
IOperand _destination;
|
||||
IOperand _lhs;
|
||||
IOperand _rhs;
|
||||
|
||||
public OpAdd(IOperand destination, IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_destination = destination;
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
_destination.Set((T)((dynamic)_lhs.Get<T>() + (dynamic)_rhs.Get<T>()));
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Operations/OpAnd.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Operations/OpAnd.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpAnd<T> : IOperation where T : unmanaged
|
||||
{
|
||||
IOperand _destination;
|
||||
IOperand _lhs;
|
||||
IOperand _rhs;
|
||||
|
||||
public OpAnd(IOperand destination, IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_destination = destination;
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
_destination.Set((T)((dynamic)_lhs.Get<T>() & (dynamic)_rhs.Get<T>()));
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Operations/OpLog.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Operations/OpLog.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
using Ryujinx.Common.Logging;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpLog<T> : IOperation where T : unmanaged
|
||||
{
|
||||
int _logId;
|
||||
IOperand _source;
|
||||
|
||||
public OpLog(int logId, IOperand source)
|
||||
{
|
||||
_logId = logId;
|
||||
_source = source;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, $"Tamper debug log id={_logId} value={(dynamic)_source.Get<T>():X}");
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Operations/OpLsh.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Operations/OpLsh.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpLsh<T> : IOperation where T : unmanaged
|
||||
{
|
||||
IOperand _destination;
|
||||
IOperand _lhs;
|
||||
IOperand _rhs;
|
||||
|
||||
public OpLsh(IOperand destination, IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_destination = destination;
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
_destination.Set((T)((dynamic)_lhs.Get<T>() << (dynamic)_rhs.Get<T>()));
|
||||
}
|
||||
}
|
||||
}
|
19
Ryujinx.HLE/HOS/Tamper/Operations/OpMov.cs
Normal file
19
Ryujinx.HLE/HOS/Tamper/Operations/OpMov.cs
Normal file
|
@ -0,0 +1,19 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpMov<T> : IOperation where T : unmanaged
|
||||
{
|
||||
IOperand _destination;
|
||||
IOperand _source;
|
||||
|
||||
public OpMov(IOperand destination, IOperand source)
|
||||
{
|
||||
_destination = destination;
|
||||
_source = source;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
_destination.Set(_source.Get<T>());
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Operations/OpMul.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Operations/OpMul.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpMul<T> : IOperation where T : unmanaged
|
||||
{
|
||||
IOperand _destination;
|
||||
IOperand _lhs;
|
||||
IOperand _rhs;
|
||||
|
||||
public OpMul(IOperand destination, IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_destination = destination;
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
_destination.Set((T)((dynamic)_lhs.Get<T>() * (dynamic)_rhs.Get<T>()));
|
||||
}
|
||||
}
|
||||
}
|
19
Ryujinx.HLE/HOS/Tamper/Operations/OpNot.cs
Normal file
19
Ryujinx.HLE/HOS/Tamper/Operations/OpNot.cs
Normal file
|
@ -0,0 +1,19 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpNot<T> : IOperation where T : unmanaged
|
||||
{
|
||||
IOperand _destination;
|
||||
IOperand _source;
|
||||
|
||||
public OpNot(IOperand destination, IOperand source)
|
||||
{
|
||||
_destination = destination;
|
||||
_source = source;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
_destination.Set((T)(~(dynamic)_source.Get<T>()));
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Operations/OpOr.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Operations/OpOr.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpOr<T> : IOperation where T : unmanaged
|
||||
{
|
||||
IOperand _destination;
|
||||
IOperand _lhs;
|
||||
IOperand _rhs;
|
||||
|
||||
public OpOr(IOperand destination, IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_destination = destination;
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
_destination.Set((T)((dynamic)_lhs.Get<T>() | (dynamic)_rhs.Get<T>()));
|
||||
}
|
||||
}
|
||||
}
|
26
Ryujinx.HLE/HOS/Tamper/Operations/OpProcCtrl.cs
Normal file
26
Ryujinx.HLE/HOS/Tamper/Operations/OpProcCtrl.cs
Normal file
|
@ -0,0 +1,26 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpProcCtrl : IOperation
|
||||
{
|
||||
private ITamperedProcess _process;
|
||||
private bool _pause;
|
||||
|
||||
public OpProcCtrl(ITamperedProcess process, bool pause)
|
||||
{
|
||||
_process = process;
|
||||
_pause = pause;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
if (_pause)
|
||||
{
|
||||
_process.PauseProcess();
|
||||
}
|
||||
else
|
||||
{
|
||||
_process.ResumeProcess();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Operations/OpRsh.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Operations/OpRsh.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpRsh<T> : IOperation where T : unmanaged
|
||||
{
|
||||
IOperand _destination;
|
||||
IOperand _lhs;
|
||||
IOperand _rhs;
|
||||
|
||||
public OpRsh(IOperand destination, IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_destination = destination;
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
_destination.Set((T)((dynamic)_lhs.Get<T>() >> (dynamic)_rhs.Get<T>()));
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Operations/OpSub.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Operations/OpSub.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpSub<T> : IOperation where T : unmanaged
|
||||
{
|
||||
IOperand _destination;
|
||||
IOperand _lhs;
|
||||
IOperand _rhs;
|
||||
|
||||
public OpSub(IOperand destination, IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_destination = destination;
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
_destination.Set((T)((dynamic)_lhs.Get<T>() - (dynamic)_rhs.Get<T>()));
|
||||
}
|
||||
}
|
||||
}
|
21
Ryujinx.HLE/HOS/Tamper/Operations/OpXor.cs
Normal file
21
Ryujinx.HLE/HOS/Tamper/Operations/OpXor.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper.Operations
|
||||
{
|
||||
class OpXor<T> : IOperation where T : unmanaged
|
||||
{
|
||||
IOperand _destination;
|
||||
IOperand _lhs;
|
||||
IOperand _rhs;
|
||||
|
||||
public OpXor(IOperand destination, IOperand lhs, IOperand rhs)
|
||||
{
|
||||
_destination = destination;
|
||||
_lhs = lhs;
|
||||
_rhs = rhs;
|
||||
}
|
||||
|
||||
public void Execute()
|
||||
{
|
||||
_destination.Set((T)((dynamic)_lhs.Get<T>() ^ (dynamic)_rhs.Get<T>()));
|
||||
}
|
||||
}
|
||||
}
|
12
Ryujinx.HLE/HOS/Tamper/Parameter.cs
Normal file
12
Ryujinx.HLE/HOS/Tamper/Parameter.cs
Normal file
|
@ -0,0 +1,12 @@
|
|||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
class Parameter<T>
|
||||
{
|
||||
public T Value { get; set; }
|
||||
|
||||
public Parameter(T value)
|
||||
{
|
||||
Value = value;
|
||||
}
|
||||
}
|
||||
}
|
32
Ryujinx.HLE/HOS/Tamper/Pointer.cs
Normal file
32
Ryujinx.HLE/HOS/Tamper/Pointer.cs
Normal file
|
@ -0,0 +1,32 @@
|
|||
using Ryujinx.Common.Logging;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
using System.Runtime.CompilerServices;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
class Pointer : IOperand
|
||||
{
|
||||
private IOperand _position;
|
||||
private ITamperedProcess _process;
|
||||
|
||||
public Pointer(IOperand position, ITamperedProcess process)
|
||||
{
|
||||
_position = position;
|
||||
_process = process;
|
||||
}
|
||||
|
||||
public T Get<T>() where T : unmanaged
|
||||
{
|
||||
return _process.ReadMemory<T>(_position.Get<ulong>());
|
||||
}
|
||||
|
||||
public void Set<T>(T value) where T : unmanaged
|
||||
{
|
||||
ulong position = _position.Get<ulong>();
|
||||
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, $"0x{position:X16}@{Unsafe.SizeOf<T>()}: {value:X}");
|
||||
|
||||
_process.WriteMemory(position, value);
|
||||
}
|
||||
}
|
||||
}
|
28
Ryujinx.HLE/HOS/Tamper/Register.cs
Normal file
28
Ryujinx.HLE/HOS/Tamper/Register.cs
Normal file
|
@ -0,0 +1,28 @@
|
|||
using Ryujinx.Common.Logging;
|
||||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
class Register : IOperand
|
||||
{
|
||||
private ulong _register = 0;
|
||||
private string _alias;
|
||||
|
||||
public Register(string alias)
|
||||
{
|
||||
_alias = alias;
|
||||
}
|
||||
|
||||
public T Get<T>() where T : unmanaged
|
||||
{
|
||||
return (T)(dynamic)_register;
|
||||
}
|
||||
|
||||
public void Set<T>(T value) where T : unmanaged
|
||||
{
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, $"{_alias}: {value}");
|
||||
|
||||
_register = (ulong)(dynamic)value;
|
||||
}
|
||||
}
|
||||
}
|
66
Ryujinx.HLE/HOS/Tamper/TamperedKProcess.cs
Normal file
66
Ryujinx.HLE/HOS/Tamper/TamperedKProcess.cs
Normal file
|
@ -0,0 +1,66 @@
|
|||
using Ryujinx.Common.Logging;
|
||||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Kernel.Process;
|
||||
using System.Runtime.CompilerServices;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
class TamperedKProcess : ITamperedProcess
|
||||
{
|
||||
private KProcess _process;
|
||||
|
||||
public ProcessState State => _process.State;
|
||||
|
||||
public TamperedKProcess(KProcess process)
|
||||
{
|
||||
this._process = process;
|
||||
}
|
||||
|
||||
private void AssertMemoryRegion<T>(ulong va, bool isWrite) where T : unmanaged
|
||||
{
|
||||
ulong size = (ulong)Unsafe.SizeOf<T>();
|
||||
|
||||
// TODO (Caian): This double check is workaround because CpuMemory.IsRangeMapped reports
|
||||
// some addresses as mapped even though they are not, i. e. 4 bytes from 0xffffffffffffff70.
|
||||
if (!_process.CpuMemory.IsMapped(va) || !_process.CpuMemory.IsRangeMapped(va, size))
|
||||
{
|
||||
throw new TamperExecutionException($"Unmapped memory access of {size} bytes at 0x{va:X16}");
|
||||
}
|
||||
|
||||
if (!isWrite)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
// TODO (Caian): It is unknown how PPTC behaves if the tamper modifies memory regions
|
||||
// belonging to code. So for now just prevent code tampering.
|
||||
if ((va >= _process.MemoryManager.CodeRegionStart) && (va + size <= _process.MemoryManager.CodeRegionEnd))
|
||||
{
|
||||
throw new CodeRegionTamperedException($"Writing {size} bytes to address 0x{va:X16} alters code");
|
||||
}
|
||||
}
|
||||
|
||||
public T ReadMemory<T>(ulong va) where T : unmanaged
|
||||
{
|
||||
AssertMemoryRegion<T>(va, false);
|
||||
|
||||
return _process.CpuMemory.Read<T>(va);
|
||||
}
|
||||
|
||||
public void WriteMemory<T>(ulong va, T value) where T : unmanaged
|
||||
{
|
||||
AssertMemoryRegion<T>(va, true);
|
||||
_process.CpuMemory.Write(va, value);
|
||||
}
|
||||
|
||||
public void PauseProcess()
|
||||
{
|
||||
Logger.Warning?.Print(LogClass.TamperMachine, "Process pausing is not supported!");
|
||||
}
|
||||
|
||||
public void ResumeProcess()
|
||||
{
|
||||
Logger.Warning?.Print(LogClass.TamperMachine, "Process resuming is not supported!");
|
||||
}
|
||||
}
|
||||
}
|
24
Ryujinx.HLE/HOS/Tamper/Value.cs
Normal file
24
Ryujinx.HLE/HOS/Tamper/Value.cs
Normal file
|
@ -0,0 +1,24 @@
|
|||
using Ryujinx.HLE.HOS.Tamper.Operations;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Tamper
|
||||
{
|
||||
class Value<P> : IOperand where P : unmanaged
|
||||
{
|
||||
private P _value;
|
||||
|
||||
public Value(P value)
|
||||
{
|
||||
_value = value;
|
||||
}
|
||||
|
||||
public T Get<T>() where T : unmanaged
|
||||
{
|
||||
return (T)(dynamic)_value;
|
||||
}
|
||||
|
||||
public void Set<T>(T value) where T : unmanaged
|
||||
{
|
||||
_value = (P)(dynamic)value;
|
||||
}
|
||||
}
|
||||
}
|
161
Ryujinx.HLE/HOS/TamperMachine.cs
Normal file
161
Ryujinx.HLE/HOS/TamperMachine.cs
Normal file
|
@ -0,0 +1,161 @@
|
|||
using Ryujinx.Common.Logging;
|
||||
using Ryujinx.HLE.Exceptions;
|
||||
using Ryujinx.HLE.HOS.Kernel;
|
||||
using Ryujinx.HLE.HOS.Kernel.Process;
|
||||
using Ryujinx.HLE.HOS.Services.Hid;
|
||||
using Ryujinx.HLE.HOS.Tamper;
|
||||
using System;
|
||||
using System.Collections.Concurrent;
|
||||
using System.Collections.Generic;
|
||||
using System.Threading;
|
||||
|
||||
namespace Ryujinx.HLE.HOS
|
||||
{
|
||||
public class TamperMachine
|
||||
{
|
||||
private Thread _tamperThread = null;
|
||||
private ConcurrentQueue<ITamperProgram> _programs = new ConcurrentQueue<ITamperProgram>();
|
||||
private long _pressedKeys = 0;
|
||||
|
||||
private void Activate()
|
||||
{
|
||||
if (_tamperThread == null || !_tamperThread.IsAlive)
|
||||
{
|
||||
_tamperThread = new Thread(this.TamperRunner);
|
||||
_tamperThread.Name = "HLE.TamperMachine";
|
||||
_tamperThread.Start();
|
||||
}
|
||||
}
|
||||
|
||||
internal void InstallAtmosphereCheat(IEnumerable<string> rawInstructions, ProcessTamperInfo info, ulong exeAddress)
|
||||
{
|
||||
if (!CanInstallOnPid(info.Process.Pid))
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
ITamperedProcess tamperedProcess = new TamperedKProcess(info.Process);
|
||||
AtmosphereCompiler compiler = new AtmosphereCompiler();
|
||||
ITamperProgram program = compiler.Compile(rawInstructions, exeAddress, info.HeapAddress, tamperedProcess);
|
||||
|
||||
if (program != null)
|
||||
{
|
||||
_programs.Enqueue(program);
|
||||
}
|
||||
|
||||
Activate();
|
||||
}
|
||||
|
||||
private bool CanInstallOnPid(long pid)
|
||||
{
|
||||
// Do not allow tampering of kernel processes.
|
||||
if (pid < KernelConstants.InitialProcessId)
|
||||
{
|
||||
Logger.Warning?.Print(LogClass.TamperMachine, $"Refusing to tamper kernel process {pid}");
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
private bool IsProcessValid(ITamperedProcess process)
|
||||
{
|
||||
return process.State != ProcessState.Crashed && process.State != ProcessState.Exiting && process.State != ProcessState.Exited;
|
||||
}
|
||||
|
||||
private void TamperRunner()
|
||||
{
|
||||
Logger.Info?.Print(LogClass.TamperMachine, "TamperMachine thread running");
|
||||
|
||||
int sleepCounter = 0;
|
||||
|
||||
while (true)
|
||||
{
|
||||
// Sleep to not consume too much CPU.
|
||||
if (sleepCounter == 0)
|
||||
{
|
||||
sleepCounter = _programs.Count;
|
||||
Thread.Sleep(1);
|
||||
}
|
||||
else
|
||||
{
|
||||
sleepCounter--;
|
||||
}
|
||||
|
||||
if (!AdvanceTamperingsQueue())
|
||||
{
|
||||
// No more work to be done.
|
||||
|
||||
Logger.Info?.Print(LogClass.TamperMachine, "TamperMachine thread exiting");
|
||||
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private bool AdvanceTamperingsQueue()
|
||||
{
|
||||
if (!_programs.TryDequeue(out ITamperProgram program))
|
||||
{
|
||||
// No more programs in the queue.
|
||||
return false;
|
||||
}
|
||||
|
||||
// Check if the process is still suitable for running the tamper program.
|
||||
if (!IsProcessValid(program.Process))
|
||||
{
|
||||
// Exit without re-enqueuing the program because the process is no longer valid.
|
||||
return true;
|
||||
}
|
||||
|
||||
// Re-enqueue the tampering program because the process is still valid.
|
||||
_programs.Enqueue(program);
|
||||
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, "Running tampering program");
|
||||
|
||||
try
|
||||
{
|
||||
ControllerKeys pressedKeys = (ControllerKeys)Thread.VolatileRead(ref _pressedKeys);
|
||||
program.Execute(pressedKeys);
|
||||
}
|
||||
catch (CodeRegionTamperedException ex)
|
||||
{
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, $"Prevented tampering program from modifing code memory");
|
||||
|
||||
if (!String.IsNullOrEmpty(ex.Message))
|
||||
{
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, ex.Message);
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, $"The tampering program crashed, this can happen while the game is starting");
|
||||
|
||||
if (!String.IsNullOrEmpty(ex.Message))
|
||||
{
|
||||
Logger.Debug?.Print(LogClass.TamperMachine, ex.Message);
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public void UpdateInput(List<GamepadInput> gamepadInputs)
|
||||
{
|
||||
// Look for the input of the player one or the handheld.
|
||||
foreach (GamepadInput input in gamepadInputs)
|
||||
{
|
||||
if (input.PlayerId == PlayerIndex.Player1 || input.PlayerId == PlayerIndex.Handheld)
|
||||
{
|
||||
Thread.VolatileWrite(ref _pressedKeys, (long)input.Buttons);
|
||||
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// Clear the input because player one is not conected.
|
||||
Thread.VolatileWrite(ref _pressedKeys, 0);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -46,6 +46,8 @@ namespace Ryujinx.HLE
|
|||
|
||||
public Hid Hid { get; private set; }
|
||||
|
||||
public TamperMachine TamperMachine { get; private set; }
|
||||
|
||||
public IHostUiHandler UiHandler { get; set; }
|
||||
|
||||
public bool EnableDeviceVsync { get; set; } = true;
|
||||
|
@ -109,6 +111,8 @@ namespace Ryujinx.HLE
|
|||
Hid.InitDevices();
|
||||
|
||||
Application = new ApplicationLoader(this, fileSystem, contentManager);
|
||||
|
||||
TamperMachine = new TamperMachine();
|
||||
}
|
||||
|
||||
public void Initialize()
|
||||
|
|
|
@ -659,6 +659,7 @@ namespace Ryujinx.Ui
|
|||
|
||||
_device.Hid.Npads.Update(gamepadInputs);
|
||||
_device.Hid.Npads.UpdateSixAxis(motionInputs);
|
||||
_device.TamperMachine.UpdateInput(gamepadInputs);
|
||||
|
||||
if(_isFocused)
|
||||
{
|
||||
|
|
Loading…
Reference in a new issue