aboutsummaryrefslogtreecommitdiff
path: root/Ryujinx.HLE/HOS/Tamper/CodeEmitters/StoreConstantToAddress.cs
diff options
context:
space:
mode:
Diffstat (limited to 'Ryujinx.HLE/HOS/Tamper/CodeEmitters/StoreConstantToAddress.cs')
-rw-r--r--Ryujinx.HLE/HOS/Tamper/CodeEmitters/StoreConstantToAddress.cs41
1 files changed, 41 insertions, 0 deletions
diff --git a/Ryujinx.HLE/HOS/Tamper/CodeEmitters/StoreConstantToAddress.cs b/Ryujinx.HLE/HOS/Tamper/CodeEmitters/StoreConstantToAddress.cs
new file mode 100644
index 00000000..933646bd
--- /dev/null
+++ b/Ryujinx.HLE/HOS/Tamper/CodeEmitters/StoreConstantToAddress.cs
@@ -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);
+ }
+ }
+}