namespace X86Disassembler.X86.Handlers; /// /// Handler for TEST instructions /// public class TestHandler : InstructionHandler { /// /// Initializes a new instance of the TestHandler class /// /// The buffer containing the code to decode /// The instruction decoder that owns this handler /// The length of the buffer public TestHandler(byte[] codeBuffer, InstructionDecoder decoder, int length) : base(codeBuffer, decoder, length) { } /// /// Checks if this handler can decode the given opcode /// /// The opcode to check /// True if this handler can decode the opcode public override bool CanHandle(byte opcode) { return opcode == 0x84 || opcode == 0x85 || opcode == 0xA8 || opcode == 0xA9; } /// /// Decodes a TEST instruction /// /// The opcode of the instruction /// The instruction object to populate /// True if the instruction was successfully decoded public override bool Decode(byte opcode, Instruction instruction) { int position = Decoder.GetPosition(); if (position >= Length) { return false; } // Set the mnemonic instruction.Mnemonic = "test"; switch (opcode) { case 0x84: // TEST r/m8, r8 case 0x85: // TEST r/m32, r32 // Read the ModR/M byte var (mod, reg, rm, destOperand) = ModRMDecoder.ReadModRM(); // Determine the source register string sourceReg; if (opcode == 0x84) // 8-bit registers { sourceReg = ModRMDecoder.GetRegister8(reg); } else // 32-bit registers { sourceReg = ModRMDecoder.GetRegister32(reg); } // Set the operands instruction.Operands = $"{destOperand}, {sourceReg}"; break; case 0xA8: // TEST AL, imm8 if (position < Length) { byte imm8 = CodeBuffer[position]; Decoder.SetPosition(position + 1); instruction.Operands = $"al, 0x{imm8:X2}"; } else { instruction.Operands = "al, ???"; } break; case 0xA9: // TEST EAX, imm32 if (position + 3 < Length) { uint imm32 = BitConverter.ToUInt32(CodeBuffer, position); Decoder.SetPosition(position + 4); instruction.Operands = $"eax, 0x{imm32:X8}"; } else { instruction.Operands = "eax, ???"; } break; default: return false; } return true; } }