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); } } }