2025-04-14 22:08:50 +03:00
|
|
|
using X86Disassembler.X86.Operands;
|
|
|
|
|
2025-04-12 22:34:02 +03:00
|
|
|
namespace X86Disassembler.X86.Handlers.Mov;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Handler for MOV r32, r/m32 instruction (0x8B) and MOV r8, r/m8 instruction (0x8A)
|
|
|
|
/// </summary>
|
|
|
|
public class MovRegMemHandler : InstructionHandler
|
|
|
|
{
|
|
|
|
/// <summary>
|
|
|
|
/// Initializes a new instance of the MovRegMemHandler class
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="decoder">The instruction decoder that owns this handler</param>
|
2025-04-14 22:08:50 +03:00
|
|
|
public MovRegMemHandler(InstructionDecoder decoder)
|
|
|
|
: base(decoder)
|
2025-04-12 22:34:02 +03:00
|
|
|
{
|
|
|
|
}
|
2025-04-13 23:06:52 +03:00
|
|
|
|
2025-04-12 22:34:02 +03:00
|
|
|
/// <summary>
|
|
|
|
/// Checks if this handler can decode the given opcode
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="opcode">The opcode to check</param>
|
|
|
|
/// <returns>True if this handler can decode the opcode</returns>
|
|
|
|
public override bool CanHandle(byte opcode)
|
|
|
|
{
|
|
|
|
return opcode == 0x8A || opcode == 0x8B;
|
|
|
|
}
|
2025-04-13 23:06:52 +03:00
|
|
|
|
2025-04-12 22:34:02 +03:00
|
|
|
/// <summary>
|
|
|
|
/// Decodes a MOV r32, r/m32 or MOV r8, r/m8 instruction
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="opcode">The opcode of the instruction</param>
|
|
|
|
/// <param name="instruction">The instruction object to populate</param>
|
|
|
|
/// <returns>True if the instruction was successfully decoded</returns>
|
|
|
|
public override bool Decode(byte opcode, Instruction instruction)
|
|
|
|
{
|
2025-04-14 22:08:50 +03:00
|
|
|
// Set the instruction type
|
|
|
|
instruction.Type = InstructionType.Mov;
|
2025-04-13 23:06:52 +03:00
|
|
|
|
2025-04-14 00:17:31 +03:00
|
|
|
// Check if we have enough bytes for the ModR/M byte
|
|
|
|
if (!Decoder.CanReadByte())
|
2025-04-12 22:34:02 +03:00
|
|
|
{
|
2025-04-14 00:17:31 +03:00
|
|
|
return false;
|
2025-04-12 22:34:02 +03:00
|
|
|
}
|
2025-04-13 23:06:52 +03:00
|
|
|
|
2025-04-12 22:34:02 +03:00
|
|
|
// Determine operand size (0 = 8-bit, 1 = 32-bit)
|
2025-04-14 00:17:31 +03:00
|
|
|
int operandSize = (opcode & 0x01) != 0 ? 32 : 8;
|
2025-04-13 23:06:52 +03:00
|
|
|
|
2025-04-13 03:56:39 +03:00
|
|
|
// Use ModRMDecoder to decode the ModR/M byte
|
2025-04-14 22:08:50 +03:00
|
|
|
// For MOV r32, r/m32 (0x8B) or MOV r8, r/m8 (0x8A):
|
|
|
|
// - The reg field specifies the destination register
|
|
|
|
// - The r/m field with mod specifies the source operand (register or memory)
|
2025-04-16 01:39:23 +03:00
|
|
|
Operand sourceOperand;
|
|
|
|
Operand destinationOperand;
|
|
|
|
|
|
|
|
if (operandSize == 8)
|
|
|
|
{
|
|
|
|
// For 8-bit operands, use the 8-bit ModR/M decoder and factory methods
|
|
|
|
var (_, reg8, _, srcOperand8) = ModRMDecoder.ReadModRM8();
|
|
|
|
sourceOperand = srcOperand8;
|
|
|
|
destinationOperand = OperandFactory.CreateRegisterOperand8(reg8);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// For 32-bit operands, use the standard ModR/M decoder
|
|
|
|
var (_, regStd, _, srcOperandStd) = ModRMDecoder.ReadModRM();
|
|
|
|
sourceOperand = srcOperandStd;
|
|
|
|
destinationOperand = OperandFactory.CreateRegisterOperand(regStd, operandSize);
|
|
|
|
}
|
2025-04-14 22:08:50 +03:00
|
|
|
|
|
|
|
// Set the structured operands
|
|
|
|
instruction.StructuredOperands =
|
|
|
|
[
|
|
|
|
destinationOperand,
|
|
|
|
sourceOperand
|
|
|
|
];
|
2025-04-13 23:06:52 +03:00
|
|
|
|
2025-04-12 22:34:02 +03:00
|
|
|
return true;
|
|
|
|
}
|
2025-04-13 23:06:52 +03:00
|
|
|
}
|