mirror of
https://github.com/holub/mame
synced 2025-05-21 21:29:15 +03:00
2165 lines
71 KiB
C
2165 lines
71 KiB
C
/*
|
|
i386 Disassembler
|
|
|
|
Written by Ville Linde
|
|
*/
|
|
|
|
#include "emu.h"
|
|
|
|
enum
|
|
{
|
|
PARAM_REG = 1, /* 16 or 32-bit register */
|
|
PARAM_REG8, /* 8-bit register */
|
|
PARAM_REG16, /* 16-bit register */
|
|
PARAM_REG32, /* 32-bit register */
|
|
PARAM_REG3264, /* 32-bit or 64-bit register */
|
|
PARAM_REG2_32, /* 32-bit register */
|
|
PARAM_MMX, /* MMX register */
|
|
PARAM_XMM, /* XMM register */
|
|
PARAM_RM, /* 16 or 32-bit memory or register */
|
|
PARAM_RM8, /* 8-bit memory or register */
|
|
PARAM_RM16, /* 16-bit memory or register */
|
|
PARAM_RM32, /* 32-bit memory or register */
|
|
PARAM_RMPTR, /* 16 or 32-bit memory or register */
|
|
PARAM_RMPTR8, /* 8-bit memory or register */
|
|
PARAM_RMPTR16, /* 16-bit memory or register */
|
|
PARAM_RMPTR32, /* 32-bit memory or register */
|
|
PARAM_RMXMM, /* 32 or 64-bit memory or register */
|
|
PARAM_REGORXMM, /* 32 or 64-bit register or XMM register */
|
|
PARAM_M64, /* 64-bit memory */
|
|
PARAM_M64PTR, /* 64-bit memory */
|
|
PARAM_MMXM, /* 64-bit memory or MMX register */
|
|
PARAM_XMMM, /* 128-bit memory or XMM register */
|
|
PARAM_I4, /* 4-bit signed immediate */
|
|
PARAM_I8, /* 8-bit signed immediate */
|
|
PARAM_I16, /* 16-bit signed immediate */
|
|
PARAM_UI8, /* 8-bit unsigned immediate */
|
|
PARAM_UI16, /* 16-bit unsigned immediate */
|
|
PARAM_IMM, /* 16 or 32-bit immediate */
|
|
PARAM_IMM64, /* 16, 32 or 64-bit immediate */
|
|
PARAM_ADDR, /* 16:16 or 16:32 address */
|
|
PARAM_REL, /* 16 or 32-bit PC-relative displacement */
|
|
PARAM_REL8, /* 8-bit PC-relative displacement */
|
|
PARAM_MEM_OFFS, /* 16 or 32-bit mem offset */
|
|
PARAM_SREG, /* segment register */
|
|
PARAM_CREG, /* control register */
|
|
PARAM_DREG, /* debug register */
|
|
PARAM_TREG, /* test register */
|
|
PARAM_1, /* used by shift/rotate instructions */
|
|
PARAM_AL,
|
|
PARAM_CL,
|
|
PARAM_DL,
|
|
PARAM_BL,
|
|
PARAM_AH,
|
|
PARAM_CH,
|
|
PARAM_DH,
|
|
PARAM_BH,
|
|
PARAM_DX,
|
|
PARAM_EAX, /* EAX or AX */
|
|
PARAM_ECX, /* ECX or CX */
|
|
PARAM_EDX, /* EDX or DX */
|
|
PARAM_EBX, /* EBX or BX */
|
|
PARAM_ESP, /* ESP or SP */
|
|
PARAM_EBP, /* EBP or BP */
|
|
PARAM_ESI, /* ESI or SI */
|
|
PARAM_EDI, /* EDI or DI */
|
|
};
|
|
|
|
enum
|
|
{
|
|
MODRM = 1,
|
|
GROUP,
|
|
FPU,
|
|
OP_SIZE,
|
|
ADDR_SIZE,
|
|
TWO_BYTE,
|
|
PREFIX,
|
|
SEG_CS,
|
|
SEG_DS,
|
|
SEG_ES,
|
|
SEG_FS,
|
|
SEG_GS,
|
|
SEG_SS,
|
|
ISREX
|
|
};
|
|
|
|
#define FLAGS_MASK 0x0ff
|
|
#define VAR_NAME 0x100
|
|
#define VAR_NAME4 0x200
|
|
#define ALWAYS64 0x400
|
|
#define SPECIAL64 0x800
|
|
#define SPECIAL64_ENT(x) (SPECIAL64 | ((x) << 24))
|
|
|
|
typedef struct {
|
|
const char *mnemonic;
|
|
UINT32 flags;
|
|
UINT32 param1;
|
|
UINT32 param2;
|
|
UINT32 param3;
|
|
offs_t dasm_flags;
|
|
} I386_OPCODE;
|
|
|
|
typedef struct {
|
|
char mnemonic[32];
|
|
const I386_OPCODE *opcode;
|
|
} GROUP_OP;
|
|
|
|
static const UINT8 *opcode_ptr;
|
|
static const UINT8 *opcode_ptr_base;
|
|
|
|
static const I386_OPCODE i386_opcode_table1[256] =
|
|
{
|
|
// 0x00
|
|
{"add", MODRM, PARAM_RM8, PARAM_REG8, 0 },
|
|
{"add", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"add", MODRM, PARAM_REG8, PARAM_RM8, 0 },
|
|
{"add", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"add", 0, PARAM_AL, PARAM_UI8, 0 },
|
|
{"add", 0, PARAM_EAX, PARAM_IMM, 0 },
|
|
{"push es", 0, 0, 0, 0 },
|
|
{"pop es", 0, 0, 0, 0 },
|
|
{"or", MODRM, PARAM_RM8, PARAM_REG8, 0 },
|
|
{"or", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"or", MODRM, PARAM_REG8, PARAM_RM8, 0 },
|
|
{"or", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"or", 0, PARAM_AL, PARAM_UI8, 0 },
|
|
{"or", 0, PARAM_EAX, PARAM_IMM, 0 },
|
|
{"push cs", 0, 0, 0, 0 },
|
|
{"two_byte", TWO_BYTE, 0, 0, 0 },
|
|
// 0x10
|
|
{"adc", MODRM, PARAM_RM8, PARAM_REG8, 0 },
|
|
{"adc", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"adc", MODRM, PARAM_REG8, PARAM_RM8, 0 },
|
|
{"adc", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"adc", 0, PARAM_AL, PARAM_UI8, 0 },
|
|
{"adc", 0, PARAM_EAX, PARAM_IMM, 0 },
|
|
{"push ss", 0, 0, 0, 0 },
|
|
{"pop ss", 0, 0, 0, 0 },
|
|
{"sbb", MODRM, PARAM_RM8, PARAM_REG8, 0 },
|
|
{"sbb", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"sbb", MODRM, PARAM_REG8, PARAM_RM8, 0 },
|
|
{"sbb", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"sbb", 0, PARAM_AL, PARAM_UI8, 0 },
|
|
{"sbb", 0, PARAM_EAX, PARAM_IMM, 0 },
|
|
{"push ds", 0, 0, 0, 0 },
|
|
{"pop ds", 0, 0, 0, 0 },
|
|
// 0x20
|
|
{"and", MODRM, PARAM_RM8, PARAM_REG8, 0 },
|
|
{"and", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"and", MODRM, PARAM_REG8, PARAM_RM8, 0 },
|
|
{"and", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"and", 0, PARAM_AL, PARAM_UI8, 0 },
|
|
{"and", 0, PARAM_EAX, PARAM_IMM, 0 },
|
|
{"seg_es", SEG_ES, 0, 0, 0 },
|
|
{"daa", 0, 0, 0, 0 },
|
|
{"sub", MODRM, PARAM_RM8, PARAM_REG8, 0 },
|
|
{"sub", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"sub", MODRM, PARAM_REG8, PARAM_RM8, 0 },
|
|
{"sub", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"sub", 0, PARAM_AL, PARAM_I8, 0 },
|
|
{"sub", 0, PARAM_EAX, PARAM_IMM, 0 },
|
|
{"seg_cs", SEG_CS, 0, 0, 0 },
|
|
{"das", 0, 0, 0, 0 },
|
|
// 0x30
|
|
{"xor", MODRM, PARAM_RM8, PARAM_REG8, 0 },
|
|
{"xor", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"xor", MODRM, PARAM_REG8, PARAM_RM8, 0 },
|
|
{"xor", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"xor", 0, PARAM_AL, PARAM_UI8, 0 },
|
|
{"xor", 0, PARAM_EAX, PARAM_IMM, 0 },
|
|
{"seg_ss", SEG_SS, 0, 0, 0 },
|
|
{"aaa", 0, 0, 0, 0 },
|
|
{"cmp", MODRM, PARAM_RM8, PARAM_REG8, 0 },
|
|
{"cmp", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"cmp", MODRM, PARAM_REG8, PARAM_RM8, 0 },
|
|
{"cmp", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmp", 0, PARAM_AL, PARAM_UI8, 0 },
|
|
{"cmp", 0, PARAM_EAX, PARAM_IMM, 0 },
|
|
{"seg_ds", SEG_DS, 0, 0, 0 },
|
|
{"aas", 0, 0, 0, 0 },
|
|
// 0x40
|
|
{"inc", ISREX, PARAM_EAX, 0, 0 },
|
|
{"inc", ISREX, PARAM_ECX, 0, 0 },
|
|
{"inc", ISREX, PARAM_EDX, 0, 0 },
|
|
{"inc", ISREX, PARAM_EBX, 0, 0 },
|
|
{"inc", ISREX, PARAM_ESP, 0, 0 },
|
|
{"inc", ISREX, PARAM_EBP, 0, 0 },
|
|
{"inc", ISREX, PARAM_ESI, 0, 0 },
|
|
{"inc", ISREX, PARAM_EDI, 0, 0 },
|
|
{"dec", ISREX, PARAM_EAX, 0, 0 },
|
|
{"dec", ISREX, PARAM_ECX, 0, 0 },
|
|
{"dec", ISREX, PARAM_EDX, 0, 0 },
|
|
{"dec", ISREX, PARAM_EBX, 0, 0 },
|
|
{"dec", ISREX, PARAM_ESP, 0, 0 },
|
|
{"dec", ISREX, PARAM_EBP, 0, 0 },
|
|
{"dec", ISREX, PARAM_ESI, 0, 0 },
|
|
{"dec", ISREX, PARAM_EDI, 0, 0 },
|
|
// 0x50
|
|
{"push", ALWAYS64, PARAM_EAX, 0, 0 },
|
|
{"push", ALWAYS64, PARAM_ECX, 0, 0 },
|
|
{"push", ALWAYS64, PARAM_EDX, 0, 0 },
|
|
{"push", ALWAYS64, PARAM_EBX, 0, 0 },
|
|
{"push", ALWAYS64, PARAM_ESP, 0, 0 },
|
|
{"push", ALWAYS64, PARAM_EBP, 0, 0 },
|
|
{"push", ALWAYS64, PARAM_ESI, 0, 0 },
|
|
{"push", ALWAYS64, PARAM_EDI, 0, 0 },
|
|
{"pop", ALWAYS64, PARAM_EAX, 0, 0 },
|
|
{"pop", ALWAYS64, PARAM_ECX, 0, 0 },
|
|
{"pop", ALWAYS64, PARAM_EDX, 0, 0 },
|
|
{"pop", ALWAYS64, PARAM_EBX, 0, 0 },
|
|
{"pop", ALWAYS64, PARAM_ESP, 0, 0 },
|
|
{"pop", ALWAYS64, PARAM_EBP, 0, 0 },
|
|
{"pop", ALWAYS64, PARAM_ESI, 0, 0 },
|
|
{"pop", ALWAYS64, PARAM_EDI, 0, 0 },
|
|
// 0x60
|
|
{"pusha\0pushad\0<invalid>",VAR_NAME,0, 0, 0 },
|
|
{"popa\0popad\0<invalid>", VAR_NAME,0, 0, 0 },
|
|
{"bound", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"arpl", MODRM | SPECIAL64_ENT(0),PARAM_RM, PARAM_REG16, 0 },
|
|
{"seg_fs", SEG_FS, 0, 0, 0 },
|
|
{"seg_gs", SEG_GS, 0, 0, 0 },
|
|
{"op_size", OP_SIZE, 0, 0, 0 },
|
|
{"addr_size", ADDR_SIZE, 0, 0, 0 },
|
|
{"push", 0, PARAM_IMM, 0, 0 },
|
|
{"imul", MODRM, PARAM_REG, PARAM_RM, PARAM_IMM },
|
|
{"push", 0, PARAM_I8, 0, 0 },
|
|
{"imul", MODRM, PARAM_REG, PARAM_RM, PARAM_I8 },
|
|
{"insb", 0, 0, 0, 0 },
|
|
{"insw\0insd\0insd",VAR_NAME, 0, 0, 0 },
|
|
{"outsb", 0, 0, 0, 0 },
|
|
{"outsw\0outsd\0outsd",VAR_NAME, 0, 0, 0 },
|
|
// 0x70
|
|
{"jo", 0, PARAM_REL8, 0, 0 },
|
|
{"jno", 0, PARAM_REL8, 0, 0 },
|
|
{"jb", 0, PARAM_REL8, 0, 0 },
|
|
{"jae", 0, PARAM_REL8, 0, 0 },
|
|
{"je", 0, PARAM_REL8, 0, 0 },
|
|
{"jne", 0, PARAM_REL8, 0, 0 },
|
|
{"jbe", 0, PARAM_REL8, 0, 0 },
|
|
{"ja", 0, PARAM_REL8, 0, 0 },
|
|
{"js", 0, PARAM_REL8, 0, 0 },
|
|
{"jns", 0, PARAM_REL8, 0, 0 },
|
|
{"jp", 0, PARAM_REL8, 0, 0 },
|
|
{"jnp", 0, PARAM_REL8, 0, 0 },
|
|
{"jl", 0, PARAM_REL8, 0, 0 },
|
|
{"jge", 0, PARAM_REL8, 0, 0 },
|
|
{"jle", 0, PARAM_REL8, 0, 0 },
|
|
{"jg", 0, PARAM_REL8, 0, 0 },
|
|
// 0x80
|
|
{"group80", GROUP, 0, 0, 0 },
|
|
{"group81", GROUP, 0, 0, 0 },
|
|
{"group80", GROUP, 0, 0, 0 },
|
|
{"group83", GROUP, 0, 0, 0 },
|
|
{"test", MODRM, PARAM_RM8, PARAM_REG8, 0 },
|
|
{"test", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"xchg", MODRM, PARAM_REG8, PARAM_RM8, 0 },
|
|
{"xchg", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"mov", MODRM, PARAM_RM8, PARAM_REG8, 0 },
|
|
{"mov", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"mov", MODRM, PARAM_REG8, PARAM_RM8, 0 },
|
|
{"mov", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"mov", MODRM, PARAM_RM, PARAM_SREG, 0 },
|
|
{"lea", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"mov", MODRM, PARAM_SREG, PARAM_RM, 0 },
|
|
{"pop", MODRM, PARAM_RM, 0, 0 },
|
|
// 0x90
|
|
{"nop", 0, 0, 0, 0 },
|
|
{"xchg", 0, PARAM_EAX, PARAM_ECX, 0 },
|
|
{"xchg", 0, PARAM_EAX, PARAM_EDX, 0 },
|
|
{"xchg", 0, PARAM_EAX, PARAM_EBX, 0 },
|
|
{"xchg", 0, PARAM_EAX, PARAM_ESP, 0 },
|
|
{"xchg", 0, PARAM_EAX, PARAM_EBP, 0 },
|
|
{"xchg", 0, PARAM_EAX, PARAM_ESI, 0 },
|
|
{"xchg", 0, PARAM_EAX, PARAM_EDI, 0 },
|
|
{"cbw\0cwde\0cdqe", VAR_NAME, 0, 0, 0 },
|
|
{"cwd\0cdq\0cqo", VAR_NAME, 0, 0, 0 },
|
|
{"call", ALWAYS64, PARAM_ADDR, 0, 0, DASMFLAG_STEP_OVER},
|
|
{"wait", 0, 0, 0, 0 },
|
|
{"pushf\0pushfd\0pushfq",VAR_NAME, 0, 0, 0 },
|
|
{"popf\0popfd\0popfq",VAR_NAME, 0, 0, 0 },
|
|
{"sahf", 0, 0, 0, 0 },
|
|
{"lahf", 0, 0, 0, 0 },
|
|
// 0xa0
|
|
{"mov", 0, PARAM_AL, PARAM_MEM_OFFS, 0 },
|
|
{"mov", 0, PARAM_EAX, PARAM_MEM_OFFS, 0 },
|
|
{"mov", 0, PARAM_MEM_OFFS, PARAM_AL, 0 },
|
|
{"mov", 0, PARAM_MEM_OFFS, PARAM_EAX, 0 },
|
|
{"movsb", 0, 0, 0, 0 },
|
|
{"movsw\0movsd\0movsq",VAR_NAME, 0, 0, 0 },
|
|
{"cmpsb", 0, 0, 0, 0 },
|
|
{"cmpsw\0cmpsd\0cmpsq",VAR_NAME, 0, 0, 0 },
|
|
{"test", 0, PARAM_AL, PARAM_UI8, 0 },
|
|
{"test", 0, PARAM_EAX, PARAM_IMM, 0 },
|
|
{"stosb", 0, 0, 0, 0 },
|
|
{"stosw\0stosd\0stosq",VAR_NAME, 0, 0, 0 },
|
|
{"lodsb", 0, 0, 0, 0 },
|
|
{"lodsw\0lodsd\0lodsq",VAR_NAME, 0, 0, 0 },
|
|
{"scasb", 0, 0, 0, 0 },
|
|
{"scasw\0scasd\0scasq",VAR_NAME, 0, 0, 0 },
|
|
// 0xb0
|
|
{"mov", 0, PARAM_AL, PARAM_UI8, 0 },
|
|
{"mov", 0, PARAM_CL, PARAM_UI8, 0 },
|
|
{"mov", 0, PARAM_DL, PARAM_UI8, 0 },
|
|
{"mov", 0, PARAM_BL, PARAM_UI8, 0 },
|
|
{"mov", 0, PARAM_AH, PARAM_UI8, 0 },
|
|
{"mov", 0, PARAM_CH, PARAM_UI8, 0 },
|
|
{"mov", 0, PARAM_DH, PARAM_UI8, 0 },
|
|
{"mov", 0, PARAM_BH, PARAM_UI8, 0 },
|
|
{"mov", 0, PARAM_EAX, PARAM_IMM64, 0 },
|
|
{"mov", 0, PARAM_ECX, PARAM_IMM64, 0 },
|
|
{"mov", 0, PARAM_EDX, PARAM_IMM64, 0 },
|
|
{"mov", 0, PARAM_EBX, PARAM_IMM64, 0 },
|
|
{"mov", 0, PARAM_ESP, PARAM_IMM64, 0 },
|
|
{"mov", 0, PARAM_EBP, PARAM_IMM64, 0 },
|
|
{"mov", 0, PARAM_ESI, PARAM_IMM64, 0 },
|
|
{"mov", 0, PARAM_EDI, PARAM_IMM64, 0 },
|
|
// 0xc0
|
|
{"groupC0", GROUP, 0, 0, 0 },
|
|
{"groupC1", GROUP, 0, 0, 0 },
|
|
{"ret", 0, PARAM_I16, 0, 0, DASMFLAG_STEP_OUT},
|
|
{"ret", 0, 0, 0, 0, DASMFLAG_STEP_OUT},
|
|
{"les", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"lds", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"mov", MODRM, PARAM_RMPTR8, PARAM_UI8, 0 },
|
|
{"mov", MODRM, PARAM_RMPTR, PARAM_IMM, 0 },
|
|
{"enter", 0, PARAM_I16, PARAM_UI8, 0 },
|
|
{"leave", 0, 0, 0, 0 },
|
|
{"retf", 0, PARAM_I16, 0, 0, DASMFLAG_STEP_OUT},
|
|
{"retf", 0, 0, 0, 0, DASMFLAG_STEP_OUT},
|
|
{"int 3", 0, 0, 0, 0, DASMFLAG_STEP_OVER},
|
|
{"int", 0, PARAM_UI8, 0, 0, DASMFLAG_STEP_OVER},
|
|
{"into", 0, 0, 0, 0 },
|
|
{"iret", 0, 0, 0, 0, DASMFLAG_STEP_OUT},
|
|
// 0xd0
|
|
{"groupD0", GROUP, 0, 0, 0 },
|
|
{"groupD1", GROUP, 0, 0, 0 },
|
|
{"groupD2", GROUP, 0, 0, 0 },
|
|
{"groupD3", GROUP, 0, 0, 0 },
|
|
{"aam", 0, PARAM_I8, 0, 0 },
|
|
{"aad", 0, PARAM_I8, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"xlat", 0, 0, 0, 0 },
|
|
{"escape", FPU, 0, 0, 0 },
|
|
{"escape", FPU, 0, 0, 0 },
|
|
{"escape", FPU, 0, 0, 0 },
|
|
{"escape", FPU, 0, 0, 0 },
|
|
{"escape", FPU, 0, 0, 0 },
|
|
{"escape", FPU, 0, 0, 0 },
|
|
{"escape", FPU, 0, 0, 0 },
|
|
{"escape", FPU, 0, 0, 0 },
|
|
// 0xe0
|
|
{"loopne", 0, PARAM_REL8, 0, 0, DASMFLAG_STEP_OVER},
|
|
{"loopz", 0, PARAM_REL8, 0, 0, DASMFLAG_STEP_OVER},
|
|
{"loop", 0, PARAM_REL8, 0, 0, DASMFLAG_STEP_OVER},
|
|
{"jcxz\0jecxz\0jrcxz",VAR_NAME, PARAM_REL8, 0, 0 },
|
|
{"in", 0, PARAM_AL, PARAM_UI8, 0 },
|
|
{"in", 0, PARAM_EAX, PARAM_UI8, 0 },
|
|
{"out", 0, PARAM_UI8, PARAM_AL, 0 },
|
|
{"out", 0, PARAM_UI8, PARAM_EAX, 0 },
|
|
{"call", 0, PARAM_REL, 0, 0, DASMFLAG_STEP_OVER},
|
|
{"jmp", 0, PARAM_REL, 0, 0 },
|
|
{"jmp", 0, PARAM_ADDR, 0, 0 },
|
|
{"jmp", 0, PARAM_REL8, 0, 0 },
|
|
{"in", 0, PARAM_AL, PARAM_DX, 0 },
|
|
{"in", 0, PARAM_EAX, PARAM_DX, 0 },
|
|
{"out", 0, PARAM_DX, PARAM_AL, 0 },
|
|
{"out", 0, PARAM_DX, PARAM_EAX, 0 },
|
|
// 0xf0
|
|
{"lock", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"repne", PREFIX, 0, 0, 0 },
|
|
{"rep", PREFIX, 0, 0, 0 },
|
|
{"hlt", 0, 0, 0, 0 },
|
|
{"cmc", 0, 0, 0, 0 },
|
|
{"groupF6", GROUP, 0, 0, 0 },
|
|
{"groupF7", GROUP, 0, 0, 0 },
|
|
{"clc", 0, 0, 0, 0 },
|
|
{"stc", 0, 0, 0, 0 },
|
|
{"cli", 0, 0, 0, 0 },
|
|
{"sti", 0, 0, 0, 0 },
|
|
{"cld", 0, 0, 0, 0 },
|
|
{"std", 0, 0, 0, 0 },
|
|
{"groupFE", GROUP, 0, 0, 0 },
|
|
{"groupFF", GROUP, 0, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE x64_opcode_alt[] =
|
|
{
|
|
{"movsxd", MODRM | ALWAYS64,PARAM_REG, PARAM_RMPTR32, 0 },
|
|
};
|
|
|
|
static const I386_OPCODE i386_opcode_table2[256] =
|
|
{
|
|
// 0x00
|
|
{"group0F00", GROUP, 0, 0, 0 },
|
|
{"group0F01", GROUP, 0, 0, 0 },
|
|
{"lar", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"lsl", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"syscall", 0, 0, 0, 0 },
|
|
{"clts", 0, 0, 0, 0 },
|
|
{"sysret", 0, 0, 0, 0 },
|
|
{"invd", 0, 0, 0, 0 },
|
|
{"wbinvd", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"ud2", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
// 0x10
|
|
{"movups\0"
|
|
"movupd\0"
|
|
"movsd\0"
|
|
"movss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"movups\0"
|
|
"movupd\0"
|
|
"movsd\0"
|
|
"movss", MODRM|VAR_NAME4,PARAM_XMMM, PARAM_XMM, 0 },
|
|
{"movlps\0"
|
|
"movlpd\0"
|
|
"movddup\0"
|
|
"movsldup", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"movlps\0"
|
|
"movlpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMMM, PARAM_XMM, 0 },
|
|
{"unpcklps\0"
|
|
"unpcklpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"unpckhps\0"
|
|
"unpckhpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"movhps\0"
|
|
"movhpd\0"
|
|
"???\0"
|
|
"movshdup", MODRM|VAR_NAME4,PARAM_XMMM, PARAM_XMM, 0 },
|
|
{"movhps\0"
|
|
"movhpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"group0F18", GROUP, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
// 0x20
|
|
{"mov", MODRM, PARAM_REG2_32, PARAM_CREG, 0 },
|
|
{"mov", MODRM, PARAM_REG2_32, PARAM_DREG, 0 },
|
|
{"mov", MODRM, PARAM_CREG, PARAM_REG2_32, 0 },
|
|
{"mov", MODRM, PARAM_DREG, PARAM_REG2_32, 0 },
|
|
{"mov", MODRM, PARAM_REG2_32, PARAM_TREG, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"mov", MODRM, PARAM_TREG, PARAM_REG2_32, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"movaps\0"
|
|
"movapd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"movaps\0"
|
|
"movapd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMMM, PARAM_XMM, 0 },
|
|
{"cvtpi2ps\0"
|
|
"cvtpi2pd\0"
|
|
"cvtsi2sd\0"
|
|
"cvtsi2ss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_RMXMM, 0 },
|
|
{"movntps\0"
|
|
"movntpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMMM, PARAM_XMM, 0 },
|
|
{"cvttps2pi\0"
|
|
"cvttpd2pi\0"
|
|
"cvttsd2si\0"
|
|
"cvttss2si", MODRM|VAR_NAME4,PARAM_REGORXMM, PARAM_XMMM, 0 },
|
|
{"cvtps2pi\0"
|
|
"cvtpd2pi\0"
|
|
"cvtsd2si\0"
|
|
"cvtss2si", MODRM|VAR_NAME4,PARAM_REGORXMM, PARAM_XMMM, 0 },
|
|
{"ucomiss\0"
|
|
"ucomisd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"comiss\0"
|
|
"comisd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
// 0x30
|
|
{"wrmsr", 0, 0, 0, 0 },
|
|
{"rdtsc", 0, 0, 0, 0 },
|
|
{"rdmsr", 0, 0, 0, 0 },
|
|
{"rdpmc", 0, 0, 0, 0 },
|
|
{"sysenter", 0, 0, 0, 0 },
|
|
{"sysexit", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
// 0x40
|
|
{"cmovo", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovno", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovb", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovae", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmove", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovne", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovbe", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmova", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovs", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovns", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovpe", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovpo", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovl", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovge", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovle", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"cmovg", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
// 0x50
|
|
{"movmskps\0"
|
|
"movmskpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_REG3264, PARAM_XMMM, 0 },
|
|
{"sqrtps\0"
|
|
"sqrtpd\0"
|
|
"sqrtsd\0"
|
|
"sqrtss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"rsqrtps\0"
|
|
"???\0"
|
|
"???\0"
|
|
"rsqrtss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"rcpps\0"
|
|
"???\0"
|
|
"???\0"
|
|
"rcpss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"andps\0"
|
|
"andpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"andnps\0"
|
|
"andnpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"orps\0"
|
|
"orpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"xorps\0"
|
|
"xorpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"addps\0"
|
|
"addpd\0"
|
|
"addsd\0"
|
|
"addss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"mulps\0"
|
|
"mulpd\0"
|
|
"mulsd\0"
|
|
"mulss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"cvtps2pd\0"
|
|
"cvtpd2ps\0"
|
|
"cvtsd2ss\0"
|
|
"cvtss2sd", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"cvtdq2ps\0"
|
|
"cvtps2dq\0"
|
|
"???\0"
|
|
"cvttps2dq", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"subps\0"
|
|
"subpd\0"
|
|
"subsd\0"
|
|
"subss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"minps\0"
|
|
"minpd\0"
|
|
"minsd\0"
|
|
"minss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"divps\0"
|
|
"divpd\0"
|
|
"divsd\0"
|
|
"divss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"maxps\0"
|
|
"maxpd\0"
|
|
"maxsd\0"
|
|
"maxss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
// 0x60
|
|
{"punpcklbw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"punpcklwd", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"punpckldq", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"packsswb", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pcmpgtb", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pcmpgtw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pcmpgtd", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"packuswb", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"punpckhbw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"punpckhwd", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"punpckhdq", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"packssdw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"punpcklqdq", MODRM, PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"punpckhqdq", MODRM, PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"movd", MODRM, PARAM_MMX, PARAM_RM, 0 },
|
|
{"movq\0"
|
|
"movdqa\0"
|
|
"???\0"
|
|
"movdqu", MODRM|VAR_NAME4,PARAM_MMX, PARAM_MMXM, 0 },
|
|
// 0x70
|
|
{"pshufw\0"
|
|
"pshufd\0"
|
|
"pshuflw\0"
|
|
"pshufhw", MODRM|VAR_NAME4,PARAM_MMX, PARAM_MMXM, PARAM_I8 },
|
|
{"group0F71", GROUP, 0, 0, 0 },
|
|
{"group0F72", GROUP, 0, 0, 0 },
|
|
{"group0F73", GROUP, 0, 0, 0 },
|
|
{"pcmpeqb", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pcmpeqw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pcmpeqd", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"emms", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???\0"
|
|
"haddpd\0"
|
|
"haddps\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"???\0"
|
|
"hsubpd\0"
|
|
"hsubps\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"movd\0"
|
|
"movd\0"
|
|
"???\0"
|
|
"movq", MODRM|VAR_NAME4,PARAM_RM, PARAM_MMX, 0 },
|
|
{"movq\0"
|
|
"movdqa\0"
|
|
"???\0"
|
|
"movdqu", MODRM|VAR_NAME4,PARAM_MMXM, PARAM_MMX, 0 },
|
|
// 0x80
|
|
{"jo", 0, PARAM_REL, 0, 0 },
|
|
{"jno", 0, PARAM_REL, 0, 0 },
|
|
{"jb", 0, PARAM_REL, 0, 0 },
|
|
{"jae", 0, PARAM_REL, 0, 0 },
|
|
{"je", 0, PARAM_REL, 0, 0 },
|
|
{"jne", 0, PARAM_REL, 0, 0 },
|
|
{"jbe", 0, PARAM_REL, 0, 0 },
|
|
{"ja", 0, PARAM_REL, 0, 0 },
|
|
{"js", 0, PARAM_REL, 0, 0 },
|
|
{"jns", 0, PARAM_REL, 0, 0 },
|
|
{"jp", 0, PARAM_REL, 0, 0 },
|
|
{"jnp", 0, PARAM_REL, 0, 0 },
|
|
{"jl", 0, PARAM_REL, 0, 0 },
|
|
{"jge", 0, PARAM_REL, 0, 0 },
|
|
{"jle", 0, PARAM_REL, 0, 0 },
|
|
{"jg", 0, PARAM_REL, 0, 0 },
|
|
// 0x90
|
|
{"seto", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setno", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setb", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setae", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"sete", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setne", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setbe", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"seta", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"sets", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setns", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setp", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setnp", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setl", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setge", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setle", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
{"setg", MODRM, PARAM_RMPTR8, 0, 0 },
|
|
// 0xa0
|
|
{"push fs", 0, 0, 0, 0 },
|
|
{"pop fs", 0, 0, 0, 0 },
|
|
{"cpuid", 0, 0, 0, 0 },
|
|
{"bt", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"shld", MODRM, PARAM_RM, PARAM_REG, PARAM_I8 },
|
|
{"shld", MODRM, PARAM_RM, PARAM_REG, PARAM_CL },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"push gs", 0, 0, 0, 0 },
|
|
{"pop gs", 0, 0, 0, 0 },
|
|
{"rsm", 0, 0, 0, 0 },
|
|
{"bts", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"shrd", MODRM, PARAM_RM, PARAM_REG, PARAM_I8 },
|
|
{"shrd", MODRM, PARAM_RM, PARAM_REG, PARAM_CL },
|
|
{"group0FAE", GROUP, 0, 0, 0 },
|
|
{"imul", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
// 0xb0
|
|
{"cmpxchg", MODRM, PARAM_RM8, PARAM_REG, 0 },
|
|
{"cmpxchg", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"lss", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"btr", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"lfs", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"lgs", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"movzx", MODRM, PARAM_REG, PARAM_RMPTR8, 0 },
|
|
{"movzx", MODRM, PARAM_REG, PARAM_RMPTR16, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"group0FBA", GROUP, 0, 0, 0 },
|
|
{"btc", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"bsf", MODRM, PARAM_REG, PARAM_RM, 0 },
|
|
{"bsr", MODRM, PARAM_REG, PARAM_RM, 0, DASMFLAG_STEP_OVER},
|
|
{"movsx", MODRM, PARAM_REG, PARAM_RMPTR8, 0 },
|
|
{"movsx", MODRM, PARAM_REG, PARAM_RMPTR16, 0 },
|
|
// 0xc0
|
|
{"xadd", MODRM, PARAM_RM8, PARAM_REG, 0 },
|
|
{"xadd", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"cmpps\0"
|
|
"cmppd\0"
|
|
"cmpsd\0"
|
|
"cmpss", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"movnti", MODRM, PARAM_RM, PARAM_REG, 0 },
|
|
{"pinsrw", MODRM, PARAM_MMX, PARAM_MMXM, PARAM_I8 },
|
|
{"pextrw", MODRM, PARAM_MMX, PARAM_MMXM, PARAM_I8 },
|
|
{"shufps\0"
|
|
"shufpd\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, PARAM_I8 },
|
|
{"cmpxchg8b", MODRM, PARAM_M64PTR, 0, 0 },
|
|
{"bswap", 0, PARAM_EAX, 0, 0 },
|
|
{"bswap", 0, PARAM_ECX, 0, 0 },
|
|
{"bswap", 0, PARAM_EDX, 0, 0 },
|
|
{"bswap", 0, PARAM_EBX, 0, 0 },
|
|
{"bswap", 0, PARAM_ESP, 0, 0 },
|
|
{"bswap", 0, PARAM_EBP, 0, 0 },
|
|
{"bswap", 0, PARAM_ESI, 0, 0 },
|
|
{"bswap", 0, PARAM_EDI, 0, 0 },
|
|
// 0xd0
|
|
{"addsubpd\0"
|
|
"???\0"
|
|
"addsubps\0"
|
|
"???\0", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"psrlw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psrld", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psrlq", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"paddq", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pmullw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"movq\0"
|
|
"???\0"
|
|
"movdq2q\0"
|
|
"movq2dq", MODRM|VAR_NAME4,PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pmovmskb", MODRM, PARAM_REG3264, PARAM_MMXM, 0 },
|
|
{"psubusb", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psubusw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pminub", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pand", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"paddub", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"padduw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pmaxub", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pandn", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
// 0xe0
|
|
{"pavgb", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psraw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psrad", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pavgw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pmulhuw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pmulhw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"???\0"
|
|
"cvttpd2dq\0"
|
|
"cvtpd2dq\0"
|
|
"cvtdq2pd", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"movntq", MODRM, PARAM_M64, PARAM_MMX, 0 },
|
|
{"psubsb", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psubsw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pminsw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"por", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"paddsb", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"paddsw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pmaxsw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pxor", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
// 0xf0
|
|
{"???\0"
|
|
"???\0"
|
|
"lddqu\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_XMM, PARAM_XMMM, 0 },
|
|
{"psllw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pslld", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psllq", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pmuludq", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"pmaddwd", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psadbw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"maskmovq\0"
|
|
"maskmovdqu\0"
|
|
"???\0"
|
|
"???", MODRM|VAR_NAME4,PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psubb", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psubw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psubd", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"psubq", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"paddb", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"paddw", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"paddd", MODRM, PARAM_MMX, PARAM_MMXM, 0 },
|
|
{"???", 0, 0, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE group80_table[8] =
|
|
{
|
|
{"add", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"or", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"adc", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"sbb", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"and", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"sub", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"xor", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"cmp", 0, PARAM_RMPTR8, PARAM_I8, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE group81_table[8] =
|
|
{
|
|
{"add", 0, PARAM_RMPTR, PARAM_IMM, 0 },
|
|
{"or", 0, PARAM_RMPTR, PARAM_IMM, 0 },
|
|
{"adc", 0, PARAM_RMPTR, PARAM_IMM, 0 },
|
|
{"sbb", 0, PARAM_RMPTR, PARAM_IMM, 0 },
|
|
{"and", 0, PARAM_RMPTR, PARAM_IMM, 0 },
|
|
{"sub", 0, PARAM_RMPTR, PARAM_IMM, 0 },
|
|
{"xor", 0, PARAM_RMPTR, PARAM_IMM, 0 },
|
|
{"cmp", 0, PARAM_RMPTR, PARAM_IMM, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE group83_table[8] =
|
|
{
|
|
{"add", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"or", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"adc", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"sbb", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"and", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"sub", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"xor", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"cmp", 0, PARAM_RMPTR, PARAM_I8, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE groupC0_table[8] =
|
|
{
|
|
{"rol", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"ror", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"rcl", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"rcr", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"shl", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"shr", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"sal", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"sar", 0, PARAM_RMPTR8, PARAM_I8, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE groupC1_table[8] =
|
|
{
|
|
{"rol", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"ror", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"rcl", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"rcr", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"shl", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"shr", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"sal", 0, PARAM_RMPTR, PARAM_I8, 0 },
|
|
{"sar", 0, PARAM_RMPTR, PARAM_I8, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE groupD0_table[8] =
|
|
{
|
|
{"rol", 0, PARAM_RMPTR8, PARAM_1, 0 },
|
|
{"ror", 0, PARAM_RMPTR8, PARAM_1, 0 },
|
|
{"rcl", 0, PARAM_RMPTR8, PARAM_1, 0 },
|
|
{"rcr", 0, PARAM_RMPTR8, PARAM_1, 0 },
|
|
{"shl", 0, PARAM_RMPTR8, PARAM_1, 0 },
|
|
{"shr", 0, PARAM_RMPTR8, PARAM_1, 0 },
|
|
{"sal", 0, PARAM_RMPTR8, PARAM_1, 0 },
|
|
{"sar", 0, PARAM_RMPTR8, PARAM_1, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE groupD1_table[8] =
|
|
{
|
|
{"rol", 0, PARAM_RMPTR, PARAM_1, 0 },
|
|
{"ror", 0, PARAM_RMPTR, PARAM_1, 0 },
|
|
{"rcl", 0, PARAM_RMPTR, PARAM_1, 0 },
|
|
{"rcr", 0, PARAM_RMPTR, PARAM_1, 0 },
|
|
{"shl", 0, PARAM_RMPTR, PARAM_1, 0 },
|
|
{"shr", 0, PARAM_RMPTR, PARAM_1, 0 },
|
|
{"sal", 0, PARAM_RMPTR, PARAM_1, 0 },
|
|
{"sar", 0, PARAM_RMPTR, PARAM_1, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE groupD2_table[8] =
|
|
{
|
|
{"rol", 0, PARAM_RMPTR8, PARAM_CL, 0 },
|
|
{"ror", 0, PARAM_RMPTR8, PARAM_CL, 0 },
|
|
{"rcl", 0, PARAM_RMPTR8, PARAM_CL, 0 },
|
|
{"rcr", 0, PARAM_RMPTR8, PARAM_CL, 0 },
|
|
{"shl", 0, PARAM_RMPTR8, PARAM_CL, 0 },
|
|
{"shr", 0, PARAM_RMPTR8, PARAM_CL, 0 },
|
|
{"sal", 0, PARAM_RMPTR8, PARAM_CL, 0 },
|
|
{"sar", 0, PARAM_RMPTR8, PARAM_CL, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE groupD3_table[8] =
|
|
{
|
|
{"rol", 0, PARAM_RMPTR, PARAM_CL, 0 },
|
|
{"ror", 0, PARAM_RMPTR, PARAM_CL, 0 },
|
|
{"rcl", 0, PARAM_RMPTR, PARAM_CL, 0 },
|
|
{"rcr", 0, PARAM_RMPTR, PARAM_CL, 0 },
|
|
{"shl", 0, PARAM_RMPTR, PARAM_CL, 0 },
|
|
{"shr", 0, PARAM_RMPTR, PARAM_CL, 0 },
|
|
{"sal", 0, PARAM_RMPTR, PARAM_CL, 0 },
|
|
{"sar", 0, PARAM_RMPTR, PARAM_CL, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE groupF6_table[8] =
|
|
{
|
|
{"test", 0, PARAM_RMPTR8, PARAM_I8, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"not", 0, PARAM_RMPTR8, 0, 0 },
|
|
{"neg", 0, PARAM_RMPTR8, 0, 0 },
|
|
{"mul", 0, PARAM_RMPTR8, 0, 0 },
|
|
{"imul", 0, PARAM_RMPTR8, 0, 0 },
|
|
{"div", 0, PARAM_RMPTR8, 0, 0 },
|
|
{"idiv", 0, PARAM_RMPTR8, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE groupF7_table[8] =
|
|
{
|
|
{"test", 0, PARAM_RMPTR, PARAM_IMM, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"not", 0, PARAM_RMPTR, 0, 0 },
|
|
{"neg", 0, PARAM_RMPTR, 0, 0 },
|
|
{"mul", 0, PARAM_RMPTR, 0, 0 },
|
|
{"imul", 0, PARAM_RMPTR, 0, 0 },
|
|
{"div", 0, PARAM_RMPTR, 0, 0 },
|
|
{"idiv", 0, PARAM_RMPTR, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE groupFE_table[8] =
|
|
{
|
|
{"inc", 0, PARAM_RMPTR8, 0, 0 },
|
|
{"dec", 0, PARAM_RMPTR8, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE groupFF_table[8] =
|
|
{
|
|
{"inc", 0, PARAM_RMPTR, 0, 0 },
|
|
{"dec", 0, PARAM_RMPTR, 0, 0 },
|
|
{"call", ALWAYS64, PARAM_RMPTR, 0, 0, DASMFLAG_STEP_OVER},
|
|
{"call far ptr ",0, PARAM_RM, 0, 0, DASMFLAG_STEP_OVER},
|
|
{"jmp", ALWAYS64, PARAM_RMPTR, 0, 0 },
|
|
{"jmp far ptr ",0, PARAM_RM, 0, 0 },
|
|
{"push", 0, PARAM_RMPTR, 0, 0 },
|
|
{"???", 0, 0, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE group0F00_table[8] =
|
|
{
|
|
{"sldt", 0, PARAM_RM, 0, 0 },
|
|
{"str", 0, PARAM_RM, 0, 0 },
|
|
{"lldt", 0, PARAM_RM, 0, 0 },
|
|
{"ltr", 0, PARAM_RM, 0, 0 },
|
|
{"verr", 0, PARAM_RM, 0, 0 },
|
|
{"verw", 0, PARAM_RM, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE group0F01_table[8] =
|
|
{
|
|
{"sgdt", 0, PARAM_RM, 0, 0 },
|
|
{"sidt", 0, PARAM_RM, 0, 0 },
|
|
{"lgdt", 0, PARAM_RM, 0, 0 },
|
|
{"lidt", 0, PARAM_RM, 0, 0 },
|
|
{"smsw", 0, PARAM_RM, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"lmsw", 0, PARAM_RM, 0, 0 },
|
|
{"invlpg", 0, PARAM_RM, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE group0F18_table[8] =
|
|
{
|
|
{"prefetchnta", 0, PARAM_RM8, 0, 0 },
|
|
{"prefetch0", 0, PARAM_RM8, 0, 0 },
|
|
{"prefetch1", 0, PARAM_RM8, 0, 0 },
|
|
{"prefetch2", 0, PARAM_RM8, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE group0F71_table[8] =
|
|
{
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"psrlw", 0, PARAM_MMX, PARAM_I8, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"psraw", 0, PARAM_MMX, PARAM_I8, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"psllw", 0, PARAM_MMX, PARAM_I8, 0 },
|
|
{"???", 0, 0, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE group0F72_table[8] =
|
|
{
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"psrld", 0, PARAM_MMX, PARAM_I8, 0 },
|
|
{"psrldq", 0, PARAM_MMX, PARAM_I8, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"pslld", 0, PARAM_MMX, PARAM_I8, 0 },
|
|
{"pslldq", 0, PARAM_MMX, PARAM_I8, 0 },
|
|
};
|
|
|
|
static const I386_OPCODE group0F73_table[8] =
|
|
{
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"psrlq", 0, PARAM_MMX, PARAM_I8, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"psraq", 0, PARAM_MMX, PARAM_I8, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"psllq", 0, PARAM_MMX, PARAM_I8, 0 },
|
|
{"???", 0, 0, 0, 0 }
|
|
};
|
|
|
|
static const I386_OPCODE group0FAE_table[8] =
|
|
{
|
|
{"fxsave", 0, PARAM_RM, 0, 0 },
|
|
{"fxrstor", 0, PARAM_RM, 0, 0 },
|
|
{"ldmxcsr", 0, PARAM_RM, 0, 0 },
|
|
{"stmxscr", 0, PARAM_RM, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"lfence", 0, 0, 0, 0 },
|
|
{"mfence", 0, 0, 0, 0 },
|
|
{"sfence", 0, 0, 0, 0 }
|
|
};
|
|
|
|
|
|
static const I386_OPCODE group0FBA_table[8] =
|
|
{
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"???", 0, 0, 0, 0 },
|
|
{"bt", 0, PARAM_RM, PARAM_I8, 0 },
|
|
{"bts", 0, PARAM_RM, PARAM_I8, 0 },
|
|
{"btr", 0, PARAM_RM, PARAM_I8, 0 },
|
|
{"btc", 0, PARAM_RM, PARAM_I8, 0 }
|
|
};
|
|
|
|
static const GROUP_OP group_op_table[] =
|
|
{
|
|
{ "group80", group80_table },
|
|
{ "group81", group81_table },
|
|
{ "group83", group83_table },
|
|
{ "groupC0", groupC0_table },
|
|
{ "groupC1", groupC1_table },
|
|
{ "groupD0", groupD0_table },
|
|
{ "groupD1", groupD1_table },
|
|
{ "groupD2", groupD2_table },
|
|
{ "groupD3", groupD3_table },
|
|
{ "groupF6", groupF6_table },
|
|
{ "groupF7", groupF7_table },
|
|
{ "groupFE", groupFE_table },
|
|
{ "groupFF", groupFF_table },
|
|
{ "group0F00", group0F00_table },
|
|
{ "group0F01", group0F01_table },
|
|
{ "group0F18", group0F18_table },
|
|
{ "group0F71", group0F71_table },
|
|
{ "group0F72", group0F72_table },
|
|
{ "group0F73", group0F73_table },
|
|
{ "group0FAE", group0FAE_table },
|
|
{ "group0FBA", group0FBA_table }
|
|
};
|
|
|
|
|
|
|
|
static const char *const i386_reg[3][16] =
|
|
{
|
|
{"ax", "cx", "dx", "bx", "sp", "bp", "si", "di", "r8w", "r9w", "r10w","r11w","r12w","r13w","r14w","r15w"},
|
|
{"eax", "ecx", "edx", "ebx", "esp", "ebp", "esi", "edi", "r8d", "r9d", "r10d","r11d","r12d","r13d","r14d","r15d"},
|
|
{"rax", "rcx", "rdx", "rbx", "rsp", "rbp", "rsi", "rdi", "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15"}
|
|
};
|
|
|
|
static const char *const i386_reg8[8] = {"al", "cl", "dl", "bl", "ah", "ch", "dh", "bh"};
|
|
static const char *const i386_reg8rex[16] = {"al", "cl", "dl", "bl", "spl", "bpl", "sil", "dil", "r8l", "r9l", "r10l", "r11l", "r12l", "r13l", "r14l", "r15l"};
|
|
static const char *const i386_sreg[8] = {"es", "cs", "ss", "ds", "fs", "gs", "???", "???"};
|
|
|
|
static int address_size;
|
|
static int operand_size;
|
|
static UINT64 pc;
|
|
static UINT8 modrm;
|
|
static UINT32 segment;
|
|
static offs_t dasm_flags;
|
|
static char modrm_string[256];
|
|
static UINT8 rex, regex, sibex, rmex;
|
|
static UINT8 pre0f;
|
|
static UINT8 curmode;
|
|
|
|
#define MODRM_REG1 ((modrm >> 3) & 0x7)
|
|
#define MODRM_REG2 (modrm & 0x7)
|
|
|
|
INLINE UINT8 FETCH(void)
|
|
{
|
|
if ((opcode_ptr - opcode_ptr_base) + 1 > 15)
|
|
return 0xff;
|
|
pc++;
|
|
return *opcode_ptr++;
|
|
}
|
|
|
|
INLINE UINT16 FETCH16(void)
|
|
{
|
|
UINT16 d;
|
|
if ((opcode_ptr - opcode_ptr_base) + 2 > 15)
|
|
return 0xffff;
|
|
d = opcode_ptr[0] | (opcode_ptr[1] << 8);
|
|
opcode_ptr += 2;
|
|
pc += 2;
|
|
return d;
|
|
}
|
|
|
|
INLINE UINT32 FETCH32(void)
|
|
{
|
|
UINT32 d;
|
|
if ((opcode_ptr - opcode_ptr_base) + 4 > 15)
|
|
return 0xffffffff;
|
|
d = opcode_ptr[0] | (opcode_ptr[1] << 8) | (opcode_ptr[2] << 16) | (opcode_ptr[3] << 24);
|
|
opcode_ptr += 4;
|
|
pc += 4;
|
|
return d;
|
|
}
|
|
|
|
INLINE UINT8 FETCHD(void)
|
|
{
|
|
if ((opcode_ptr - opcode_ptr_base) + 1 > 15)
|
|
return 0xff;
|
|
pc++;
|
|
return *opcode_ptr++;
|
|
}
|
|
|
|
INLINE UINT16 FETCHD16(void)
|
|
{
|
|
UINT16 d;
|
|
if ((opcode_ptr - opcode_ptr_base) + 2 > 15)
|
|
return 0xffff;
|
|
d = opcode_ptr[0] | (opcode_ptr[1] << 8);
|
|
opcode_ptr += 2;
|
|
pc += 2;
|
|
return d;
|
|
}
|
|
|
|
INLINE UINT32 FETCHD32(void)
|
|
{
|
|
UINT32 d;
|
|
if ((opcode_ptr - opcode_ptr_base) + 4 > 15)
|
|
return 0xffffffff;
|
|
d = opcode_ptr[0] | (opcode_ptr[1] << 8) | (opcode_ptr[2] << 16) | (opcode_ptr[3] << 24);
|
|
opcode_ptr += 4;
|
|
pc += 4;
|
|
return d;
|
|
}
|
|
|
|
static char *hexstring(UINT32 value, int digits)
|
|
{
|
|
static char buffer[20];
|
|
buffer[0] = '0';
|
|
if (digits)
|
|
sprintf(&buffer[1], "%0*Xh", digits, value);
|
|
else
|
|
sprintf(&buffer[1], "%Xh", value);
|
|
return (buffer[1] >= '0' && buffer[1] <= '9') ? &buffer[1] : &buffer[0];
|
|
}
|
|
|
|
static char *hexstring64(UINT32 lo, UINT32 hi)
|
|
{
|
|
static char buffer[40];
|
|
buffer[0] = '0';
|
|
if (hi != 0)
|
|
sprintf(&buffer[1], "%X%08Xh", hi, lo);
|
|
else
|
|
sprintf(&buffer[1], "%Xh", lo);
|
|
return (buffer[1] >= '0' && buffer[1] <= '9') ? &buffer[1] : &buffer[0];
|
|
}
|
|
|
|
static char *hexstringpc(UINT64 pc)
|
|
{
|
|
if (curmode == 64)
|
|
return hexstring64((UINT32)pc, (UINT32)(pc >> 32));
|
|
else
|
|
return hexstring((UINT32)pc, 0);
|
|
}
|
|
|
|
static char *shexstring(UINT32 value, int digits, int always)
|
|
{
|
|
static char buffer[20];
|
|
if (value >= 0x80000000)
|
|
sprintf(buffer, "-%s", hexstring(-value, digits));
|
|
else if (always)
|
|
sprintf(buffer, "+%s", hexstring(value, digits));
|
|
else
|
|
return hexstring(value, digits);
|
|
return buffer;
|
|
}
|
|
|
|
static char* handle_sib_byte( char* s, UINT8 mod )
|
|
{
|
|
UINT32 i32;
|
|
UINT8 scale, i, base;
|
|
UINT8 sib = FETCHD();
|
|
|
|
scale = (sib >> 6) & 0x3;
|
|
i = ((sib >> 3) & 0x7) | sibex;
|
|
base = (sib & 0x7) | rmex;
|
|
|
|
if (base == 5 && mod == 0) {
|
|
i32 = FETCH32();
|
|
s += sprintf( s, "%s", hexstring(i32, 0) );
|
|
} else if (base != 5 || mod != 3)
|
|
s += sprintf( s, "%s", i386_reg[address_size][base] );
|
|
|
|
if ( i != 4 ) {
|
|
s += sprintf( s, "+%s", i386_reg[address_size][i] );
|
|
if (scale)
|
|
s += sprintf( s, "*%d", 1 << scale );
|
|
}
|
|
return s;
|
|
}
|
|
|
|
static void handle_modrm(char* s)
|
|
{
|
|
INT8 disp8;
|
|
INT16 disp16;
|
|
INT32 disp32;
|
|
UINT8 mod, rm;
|
|
|
|
modrm = FETCHD();
|
|
mod = (modrm >> 6) & 0x3;
|
|
rm = (modrm & 0x7) | rmex;
|
|
|
|
if( modrm >= 0xc0 )
|
|
return;
|
|
|
|
switch(segment)
|
|
{
|
|
case SEG_CS: s += sprintf( s, "cs:" ); break;
|
|
case SEG_DS: s += sprintf( s, "ds:" ); break;
|
|
case SEG_ES: s += sprintf( s, "es:" ); break;
|
|
case SEG_FS: s += sprintf( s, "fs:" ); break;
|
|
case SEG_GS: s += sprintf( s, "gs:" ); break;
|
|
case SEG_SS: s += sprintf( s, "ss:" ); break;
|
|
}
|
|
|
|
s += sprintf( s, "[" );
|
|
if( address_size == 2 ) {
|
|
if ((rm & 7) == 4)
|
|
s = handle_sib_byte( s, mod );
|
|
else if ((rm & 7) == 5 && mod == 0) {
|
|
disp32 = FETCHD32();
|
|
s += sprintf( s, "rip%s", shexstring(disp32, 0, TRUE) );
|
|
} else
|
|
s += sprintf( s, "%s", i386_reg[2][rm]);
|
|
if( mod == 1 ) {
|
|
disp8 = FETCHD();
|
|
if (disp8 != 0)
|
|
s += sprintf( s, "%s", shexstring((INT32)disp8, 0, TRUE) );
|
|
} else if( mod == 2 ) {
|
|
disp32 = FETCHD32();
|
|
if (disp32 != 0)
|
|
s += sprintf( s, "%s", shexstring(disp32, 0, TRUE) );
|
|
}
|
|
} else if (address_size == 1) {
|
|
if ((rm & 7) == 4)
|
|
s = handle_sib_byte( s, mod );
|
|
else if ((rm & 7) == 5 && mod == 0) {
|
|
disp32 = FETCHD32();
|
|
if (curmode == 64)
|
|
s += sprintf( s, "eip%s", shexstring(disp32, 0, TRUE) );
|
|
else
|
|
s += sprintf( s, "%s", hexstring(disp32, 0) );
|
|
} else
|
|
s += sprintf( s, "%s", i386_reg[1][rm]);
|
|
if( mod == 1 ) {
|
|
disp8 = FETCHD();
|
|
if (disp8 != 0)
|
|
s += sprintf( s, "%s", shexstring((INT32)disp8, 0, TRUE) );
|
|
} else if( mod == 2 ) {
|
|
disp32 = FETCHD32();
|
|
if (disp32 != 0)
|
|
s += sprintf( s, "%s", shexstring(disp32, 0, TRUE) );
|
|
}
|
|
} else {
|
|
switch( rm )
|
|
{
|
|
case 0: s += sprintf( s, "bx+si" ); break;
|
|
case 1: s += sprintf( s, "bx+di" ); break;
|
|
case 2: s += sprintf( s, "bx+si" ); break;
|
|
case 3: s += sprintf( s, "bx+di" ); break;
|
|
case 4: s += sprintf( s, "si" ); break;
|
|
case 5: s += sprintf( s, "di" ); break;
|
|
case 6:
|
|
if( mod == 0 ) {
|
|
disp16 = FETCHD16();
|
|
s += sprintf( s, "%s", hexstring((unsigned) (UINT16) disp16, 0) );
|
|
} else {
|
|
s += sprintf( s, "bp" );
|
|
}
|
|
break;
|
|
case 7: s += sprintf( s, "bx" ); break;
|
|
}
|
|
if( mod == 1 ) {
|
|
disp8 = FETCHD();
|
|
if (disp8 != 0)
|
|
s += sprintf( s, "%s", shexstring((INT32)disp8, 0, TRUE) );
|
|
} else if( mod == 2 ) {
|
|
disp16 = FETCHD16();
|
|
if (disp16 != 0)
|
|
s += sprintf( s, "%s", shexstring((INT32)disp16, 0, TRUE) );
|
|
}
|
|
}
|
|
s += sprintf( s, "]" );
|
|
}
|
|
|
|
static char* handle_param(char* s, UINT32 param)
|
|
{
|
|
UINT8 i8;
|
|
UINT16 i16;
|
|
UINT32 i32;
|
|
UINT16 ptr;
|
|
UINT32 addr;
|
|
INT8 d8;
|
|
INT16 d16;
|
|
INT32 d32;
|
|
|
|
switch(param)
|
|
{
|
|
case PARAM_REG:
|
|
s += sprintf( s, "%s", i386_reg[operand_size][MODRM_REG1 | regex] );
|
|
break;
|
|
|
|
case PARAM_REG8:
|
|
s += sprintf( s, "%s", (rex ? i386_reg8rex : i386_reg8)[MODRM_REG1 | regex] );
|
|
break;
|
|
|
|
case PARAM_REG16:
|
|
s += sprintf( s, "%s", i386_reg[0][MODRM_REG1 | regex] );
|
|
break;
|
|
|
|
case PARAM_REG32:
|
|
s += sprintf( s, "%s", i386_reg[1][MODRM_REG1 | regex] );
|
|
break;
|
|
|
|
case PARAM_REG3264:
|
|
s += sprintf( s, "%s", i386_reg[(operand_size == 2) ? 2 : 1][MODRM_REG1 | regex] );
|
|
break;
|
|
|
|
case PARAM_MMX:
|
|
if (pre0f == 0x66 || pre0f == 0xf2 || pre0f == 0xf3)
|
|
s += sprintf( s, "xmm%d", MODRM_REG1 | regex );
|
|
else
|
|
s += sprintf( s, "mm%d", MODRM_REG1 | regex );
|
|
break;
|
|
|
|
case PARAM_XMM:
|
|
s += sprintf( s, "xmm%d", MODRM_REG1 | regex );
|
|
break;
|
|
|
|
case PARAM_REGORXMM:
|
|
if (pre0f != 0xf2 && pre0f != 0xf3)
|
|
s += sprintf( s, "xmm%d", MODRM_REG1 | regex );
|
|
else
|
|
s += sprintf( s, "%s", i386_reg[(operand_size == 2) ? 2 : 1][MODRM_REG1 | regex] );
|
|
break;
|
|
|
|
case PARAM_REG2_32:
|
|
s += sprintf( s, "%s", i386_reg[1][MODRM_REG2 | rmex] );
|
|
break;
|
|
|
|
case PARAM_RM:
|
|
case PARAM_RMPTR:
|
|
if( modrm >= 0xc0 ) {
|
|
s += sprintf( s, "%s", i386_reg[operand_size][MODRM_REG2 | rmex] );
|
|
} else {
|
|
if (param == PARAM_RMPTR)
|
|
{
|
|
if( operand_size == 2 )
|
|
s += sprintf( s, "qword ptr " );
|
|
else if (operand_size == 1)
|
|
s += sprintf( s, "dword ptr " );
|
|
else
|
|
s += sprintf( s, "word ptr " );
|
|
}
|
|
s += sprintf( s, "%s", modrm_string );
|
|
}
|
|
break;
|
|
|
|
case PARAM_RM8:
|
|
case PARAM_RMPTR8:
|
|
if( modrm >= 0xc0 ) {
|
|
s += sprintf( s, "%s", (rex ? i386_reg8rex : i386_reg8)[MODRM_REG2 | rmex] );
|
|
} else {
|
|
if (param == PARAM_RMPTR8)
|
|
s += sprintf( s, "byte ptr " );
|
|
s += sprintf( s, "%s", modrm_string );
|
|
}
|
|
break;
|
|
|
|
case PARAM_RM16:
|
|
case PARAM_RMPTR16:
|
|
if( modrm >= 0xc0 ) {
|
|
s += sprintf( s, "%s", i386_reg[0][MODRM_REG2 | rmex] );
|
|
} else {
|
|
if (param == PARAM_RMPTR16)
|
|
s += sprintf( s, "word ptr " );
|
|
s += sprintf( s, "%s", modrm_string );
|
|
}
|
|
break;
|
|
|
|
case PARAM_RM32:
|
|
case PARAM_RMPTR32:
|
|
if( modrm >= 0xc0 ) {
|
|
s += sprintf( s, "%s", i386_reg[1][MODRM_REG2 | rmex] );
|
|
} else {
|
|
if (param == PARAM_RMPTR32)
|
|
s += sprintf( s, "dword ptr " );
|
|
s += sprintf( s, "%s", modrm_string );
|
|
}
|
|
break;
|
|
|
|
case PARAM_RMXMM:
|
|
if( modrm >= 0xc0 ) {
|
|
if (pre0f != 0xf2 && pre0f != 0xf3)
|
|
s += sprintf( s, "xmm%d", MODRM_REG2 | rmex );
|
|
else
|
|
s += sprintf( s, "%s", i386_reg[(operand_size == 2) ? 2 : 1][MODRM_REG2 | rmex] );
|
|
} else {
|
|
if (param == PARAM_RMPTR32)
|
|
s += sprintf( s, "dword ptr " );
|
|
s += sprintf( s, "%s", modrm_string );
|
|
}
|
|
break;
|
|
|
|
case PARAM_M64:
|
|
case PARAM_M64PTR:
|
|
if( modrm >= 0xc0 ) {
|
|
s += sprintf( s, "???" );
|
|
} else {
|
|
if (param == PARAM_M64PTR)
|
|
s += sprintf( s, "qword ptr " );
|
|
s += sprintf( s, "%s", modrm_string );
|
|
}
|
|
break;
|
|
|
|
case PARAM_MMXM:
|
|
if( modrm >= 0xc0 ) {
|
|
if (pre0f == 0x66 || pre0f == 0xf2 || pre0f == 0xf3)
|
|
s += sprintf( s, "xmm%d", MODRM_REG2 | rmex );
|
|
else
|
|
s += sprintf( s, "mm%d", MODRM_REG2 | rmex );
|
|
} else {
|
|
s += sprintf( s, "%s", modrm_string );
|
|
}
|
|
break;
|
|
|
|
case PARAM_XMMM:
|
|
if( modrm >= 0xc0 ) {
|
|
s += sprintf( s, "xmm%d", MODRM_REG2 | rmex );
|
|
} else {
|
|
s += sprintf( s, "%s", modrm_string );
|
|
}
|
|
break;
|
|
|
|
case PARAM_I4:
|
|
i8 = FETCHD();
|
|
s += sprintf( s, "%d", i8 & 0x0f );
|
|
break;
|
|
|
|
case PARAM_I8:
|
|
i8 = FETCHD();
|
|
s += sprintf( s, "%s", shexstring((INT8)i8, 0, FALSE) );
|
|
break;
|
|
|
|
case PARAM_I16:
|
|
i16 = FETCHD16();
|
|
s += sprintf( s, "%s", shexstring((INT16)i16, 0, FALSE) );
|
|
break;
|
|
|
|
case PARAM_UI8:
|
|
i8 = FETCHD();
|
|
s += sprintf( s, "%s", shexstring((UINT8)i8, 0, FALSE) );
|
|
break;
|
|
|
|
case PARAM_UI16:
|
|
i16 = FETCHD16();
|
|
s += sprintf( s, "%s", shexstring((UINT16)i16, 0, FALSE) );
|
|
break;
|
|
|
|
case PARAM_IMM64:
|
|
if (operand_size == 2) {
|
|
UINT32 lo32 = FETCHD32();
|
|
i32 = FETCHD32();
|
|
s += sprintf( s, "%s", hexstring64(lo32, i32) );
|
|
} else if( operand_size ) {
|
|
i32 = FETCHD32();
|
|
s += sprintf( s, "%s", hexstring(i32, 0) );
|
|
} else {
|
|
i16 = FETCHD16();
|
|
s += sprintf( s, "%s", hexstring(i16, 0) );
|
|
}
|
|
break;
|
|
|
|
case PARAM_IMM:
|
|
if( operand_size ) {
|
|
i32 = FETCHD32();
|
|
s += sprintf( s, "%s", hexstring(i32, 0) );
|
|
} else {
|
|
i16 = FETCHD16();
|
|
s += sprintf( s, "%s", hexstring(i16, 0) );
|
|
}
|
|
break;
|
|
|
|
case PARAM_ADDR:
|
|
if( operand_size ) {
|
|
addr = FETCHD32();
|
|
ptr = FETCHD16();
|
|
s += sprintf( s, "%s:", hexstring(ptr, 4) );
|
|
s += sprintf( s, "%s", hexstring(addr, 0) );
|
|
} else {
|
|
addr = FETCHD16();
|
|
ptr = FETCHD16();
|
|
s += sprintf( s, "%s:", hexstring(ptr, 4) );
|
|
s += sprintf( s, "%s", hexstring(addr, 0) );
|
|
}
|
|
break;
|
|
|
|
case PARAM_REL:
|
|
if( operand_size ) {
|
|
d32 = FETCHD32();
|
|
s += sprintf( s, "%s", hexstringpc(pc + d32) );
|
|
} else {
|
|
/* make sure to keep the relative offset within the segment */
|
|
d16 = FETCHD16();
|
|
s += sprintf( s, "%s", hexstringpc((pc & 0xFFFF0000) | ((pc + d16) & 0x0000FFFF)) );
|
|
}
|
|
break;
|
|
|
|
case PARAM_REL8:
|
|
d8 = FETCHD();
|
|
s += sprintf( s, "%s", hexstringpc(pc + d8) );
|
|
break;
|
|
|
|
case PARAM_MEM_OFFS:
|
|
switch(segment)
|
|
{
|
|
case SEG_CS: s += sprintf( s, "cs:" ); break;
|
|
case SEG_DS: s += sprintf( s, "ds:" ); break;
|
|
case SEG_ES: s += sprintf( s, "es:" ); break;
|
|
case SEG_FS: s += sprintf( s, "fs:" ); break;
|
|
case SEG_GS: s += sprintf( s, "gs:" ); break;
|
|
case SEG_SS: s += sprintf( s, "ss:" ); break;
|
|
}
|
|
|
|
if( address_size ) {
|
|
d32 = FETCHD32();
|
|
s += sprintf( s, "[%s]", hexstring(d32, 0) );
|
|
} else {
|
|
d32 = FETCHD16();
|
|
s += sprintf( s, "[%s]", hexstring(d32, 0) );
|
|
}
|
|
break;
|
|
|
|
case PARAM_SREG:
|
|
s += sprintf( s, "%s", i386_sreg[MODRM_REG1] );
|
|
break;
|
|
|
|
case PARAM_CREG:
|
|
s += sprintf( s, "cr%d", MODRM_REG1 | regex );
|
|
break;
|
|
|
|
case PARAM_TREG:
|
|
s += sprintf( s, "tr%d", MODRM_REG1 | regex );
|
|
break;
|
|
|
|
case PARAM_DREG:
|
|
s += sprintf( s, "dr%d", MODRM_REG1 | regex );
|
|
break;
|
|
|
|
case PARAM_1:
|
|
s += sprintf( s, "1" );
|
|
break;
|
|
|
|
case PARAM_DX:
|
|
s += sprintf( s, "dx" );
|
|
break;
|
|
|
|
case PARAM_AL: s += sprintf( s, "al" ); break;
|
|
case PARAM_CL: s += sprintf( s, "cl" ); break;
|
|
case PARAM_DL: s += sprintf( s, "dl" ); break;
|
|
case PARAM_BL: s += sprintf( s, "bl" ); break;
|
|
case PARAM_AH: s += sprintf( s, "ah" ); break;
|
|
case PARAM_CH: s += sprintf( s, "ch" ); break;
|
|
case PARAM_DH: s += sprintf( s, "dh" ); break;
|
|
case PARAM_BH: s += sprintf( s, "bh" ); break;
|
|
|
|
case PARAM_EAX: s += sprintf( s, "%s", i386_reg[operand_size][0 | rmex] ); break;
|
|
case PARAM_ECX: s += sprintf( s, "%s", i386_reg[operand_size][1 | rmex] ); break;
|
|
case PARAM_EDX: s += sprintf( s, "%s", i386_reg[operand_size][2 | rmex] ); break;
|
|
case PARAM_EBX: s += sprintf( s, "%s", i386_reg[operand_size][3 | rmex] ); break;
|
|
case PARAM_ESP: s += sprintf( s, "%s", i386_reg[operand_size][4 | rmex] ); break;
|
|
case PARAM_EBP: s += sprintf( s, "%s", i386_reg[operand_size][5 | rmex] ); break;
|
|
case PARAM_ESI: s += sprintf( s, "%s", i386_reg[operand_size][6 | rmex] ); break;
|
|
case PARAM_EDI: s += sprintf( s, "%s", i386_reg[operand_size][7 | rmex] ); break;
|
|
}
|
|
return s;
|
|
}
|
|
|
|
static void handle_fpu(char *s, UINT8 op1, UINT8 op2)
|
|
{
|
|
switch (op1 & 0x7)
|
|
{
|
|
case 0: // Group D8
|
|
{
|
|
if (op2 < 0xc0)
|
|
{
|
|
pc--; // adjust fetch pointer, so modrm byte read again
|
|
opcode_ptr--;
|
|
handle_modrm( modrm_string );
|
|
switch ((op2 >> 3) & 0x7)
|
|
{
|
|
case 0: sprintf(s, "fadd dword ptr %s", modrm_string); break;
|
|
case 1: sprintf(s, "fmul dword ptr %s", modrm_string); break;
|
|
case 2: sprintf(s, "fcom dword ptr %s", modrm_string); break;
|
|
case 3: sprintf(s, "fcomp dword ptr %s", modrm_string); break;
|
|
case 4: sprintf(s, "fsub dword ptr %s", modrm_string); break;
|
|
case 5: sprintf(s, "fsubr dword ptr %s", modrm_string); break;
|
|
case 6: sprintf(s, "fdiv dword ptr %s", modrm_string); break;
|
|
case 7: sprintf(s, "fdivr dword ptr %s", modrm_string); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch ((op2 >> 3) & 0x7)
|
|
{
|
|
case 0: sprintf(s, "fadd st(0),st(%d)", op2 & 0x7); break;
|
|
case 1: sprintf(s, "fmul st(0),st(%d)", op2 & 0x7); break;
|
|
case 2: sprintf(s, "fcom st(0),st(%d)", op2 & 0x7); break;
|
|
case 3: sprintf(s, "fcomp st(0),st(%d)", op2 & 0x7); break;
|
|
case 4: sprintf(s, "fsub st(0),st(%d)", op2 & 0x7); break;
|
|
case 5: sprintf(s, "fsubr st(0),st(%d)", op2 & 0x7); break;
|
|
case 6: sprintf(s, "fdiv st(0),st(%d)", op2 & 0x7); break;
|
|
case 7: sprintf(s, "fdivr st(0),st(%d)", op2 & 0x7); break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 1: // Group D9
|
|
{
|
|
if (op2 < 0xc0)
|
|
{
|
|
pc--; // adjust fetch pointer, so modrm byte read again
|
|
opcode_ptr--;
|
|
handle_modrm( modrm_string );
|
|
switch ((op2 >> 3) & 0x7)
|
|
{
|
|
case 0: sprintf(s, "fld dword ptr %s", modrm_string); break;
|
|
case 1: sprintf(s, "??? (FPU)"); break;
|
|
case 2: sprintf(s, "fst dword ptr %s", modrm_string); break;
|
|
case 3: sprintf(s, "fstp dword ptr %s", modrm_string); break;
|
|
case 4: sprintf(s, "fldenv word ptr %s", modrm_string); break;
|
|
case 5: sprintf(s, "fldcw word ptr %s", modrm_string); break;
|
|
case 6: sprintf(s, "fstenv word ptr %s", modrm_string); break;
|
|
case 7: sprintf(s, "fstcw word ptr %s", modrm_string); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch (op2 & 0x3f)
|
|
{
|
|
case 0x00: case 0x01: case 0x02: case 0x03: case 0x04: case 0x05: case 0x06: case 0x07:
|
|
sprintf(s, "fld st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x08: case 0x09: case 0x0a: case 0x0b: case 0x0c: case 0x0d: case 0x0e: case 0x0f:
|
|
sprintf(s, "fxch st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x10: sprintf(s, "fnop"); break;
|
|
case 0x20: sprintf(s, "fchs"); break;
|
|
case 0x21: sprintf(s, "fabs"); break;
|
|
case 0x24: sprintf(s, "ftst"); break;
|
|
case 0x25: sprintf(s, "fxam"); break;
|
|
case 0x28: sprintf(s, "fld1"); break;
|
|
case 0x29: sprintf(s, "fldl2t"); break;
|
|
case 0x2a: sprintf(s, "fldl2e"); break;
|
|
case 0x2b: sprintf(s, "fldpi"); break;
|
|
case 0x2c: sprintf(s, "fldlg2"); break;
|
|
case 0x2d: sprintf(s, "fldln2"); break;
|
|
case 0x2e: sprintf(s, "fldz"); break;
|
|
case 0x30: sprintf(s, "f2xm1"); break;
|
|
case 0x31: sprintf(s, "fyl2x"); break;
|
|
case 0x32: sprintf(s, "fptan"); break;
|
|
case 0x33: sprintf(s, "fpatan"); break;
|
|
case 0x34: sprintf(s, "fxtract"); break;
|
|
case 0x35: sprintf(s, "fprem1"); break;
|
|
case 0x36: sprintf(s, "fdecstp"); break;
|
|
case 0x37: sprintf(s, "fincstp"); break;
|
|
case 0x38: sprintf(s, "fprem"); break;
|
|
case 0x39: sprintf(s, "fyl2xp1"); break;
|
|
case 0x3a: sprintf(s, "fsqrt"); break;
|
|
case 0x3b: sprintf(s, "fsincos"); break;
|
|
case 0x3c: sprintf(s, "frndint"); break;
|
|
case 0x3d: sprintf(s, "fscale"); break;
|
|
case 0x3e: sprintf(s, "fsin"); break;
|
|
case 0x3f: sprintf(s, "fcos"); break;
|
|
|
|
default: sprintf(s, "??? (FPU)"); break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 2: // Group DA
|
|
{
|
|
if (op2 < 0xc0)
|
|
{
|
|
pc--; // adjust fetch pointer, so modrm byte read again
|
|
opcode_ptr--;
|
|
handle_modrm( modrm_string );
|
|
switch ((op2 >> 3) & 0x7)
|
|
{
|
|
case 0: sprintf(s, "fiadd dword ptr %s", modrm_string); break;
|
|
case 1: sprintf(s, "fimul dword ptr %s", modrm_string); break;
|
|
case 2: sprintf(s, "ficom dword ptr %s", modrm_string); break;
|
|
case 3: sprintf(s, "ficomp dword ptr %s", modrm_string); break;
|
|
case 4: sprintf(s, "fisub dword ptr %s", modrm_string); break;
|
|
case 5: sprintf(s, "fisubr dword ptr %s", modrm_string); break;
|
|
case 6: sprintf(s, "fidiv dword ptr %s", modrm_string); break;
|
|
case 7: sprintf(s, "fidivr dword ptr %s", modrm_string); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch (op2 & 0x3f)
|
|
{
|
|
case 0x00: case 0x01: case 0x02: case 0x03: case 0x04: case 0x05: case 0x06: case 0x07:
|
|
sprintf(s, "fcmovb st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x08: case 0x09: case 0x0a: case 0x0b: case 0x0c: case 0x0d: case 0x0e: case 0x0f:
|
|
sprintf(s, "fcmove st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x10: case 0x11: case 0x12: case 0x13: case 0x14: case 0x15: case 0x16: case 0x17:
|
|
sprintf(s, "fcmovbe st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x18: case 0x19: case 0x1a: case 0x1b: case 0x1c: case 0x1d: case 0x1e: case 0x1f:
|
|
sprintf(s, "fcmovu st(0),st(%d)", op2 & 0x7); break;
|
|
case 0x29:
|
|
sprintf(s, "fucompp"); break;
|
|
|
|
default: sprintf(s, "??? (FPU)"); break;
|
|
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 3: // Group DB
|
|
{
|
|
if (op2 < 0xc0)
|
|
{
|
|
pc--; // adjust fetch pointer, so modrm byte read again
|
|
opcode_ptr--;
|
|
handle_modrm( modrm_string );
|
|
switch ((op2 >> 3) & 0x7)
|
|
{
|
|
case 0: sprintf(s, "fild dword ptr %s", modrm_string); break;
|
|
case 1: sprintf(s, "fisttp dword ptr %s", modrm_string); break;
|
|
case 2: sprintf(s, "fist dword ptr %s", modrm_string); break;
|
|
case 3: sprintf(s, "fistp dword ptr %s", modrm_string); break;
|
|
case 4: sprintf(s, "??? (FPU)"); break;
|
|
case 5: sprintf(s, "fld tword ptr %s", modrm_string); break;
|
|
case 6: sprintf(s, "??? (FPU)"); break;
|
|
case 7: sprintf(s, "fstp tword ptr %s", modrm_string); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch (op2 & 0x3f)
|
|
{
|
|
case 0x00: case 0x01: case 0x02: case 0x03: case 0x04: case 0x05: case 0x06: case 0x07:
|
|
sprintf(s, "fcmovnb st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x08: case 0x09: case 0x0a: case 0x0b: case 0x0c: case 0x0d: case 0x0e: case 0x0f:
|
|
sprintf(s, "fcmovne st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x10: case 0x11: case 0x12: case 0x13: case 0x14: case 0x15: case 0x16: case 0x17:
|
|
sprintf(s, "fcmovnbe st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x18: case 0x19: case 0x1a: case 0x1b: case 0x1c: case 0x1d: case 0x1e: case 0x1f:
|
|
sprintf(s, "fcmovnu st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x22: sprintf(s, "fclex"); break;
|
|
case 0x23: sprintf(s, "finit"); break;
|
|
|
|
case 0x28: case 0x29: case 0x2a: case 0x2b: case 0x2c: case 0x2d: case 0x2e: case 0x2f:
|
|
sprintf(s, "fucomi st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x30: case 0x31: case 0x32: case 0x33: case 0x34: case 0x35: case 0x36: case 0x37:
|
|
sprintf(s, "fcomi st(0),st(%d)", op2 & 0x7); break;
|
|
|
|
default: sprintf(s, "??? (FPU)"); break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 4: // Group DC
|
|
{
|
|
if (op2 < 0xc0)
|
|
{
|
|
pc--; // adjust fetch pointer, so modrm byte read again
|
|
opcode_ptr--;
|
|
handle_modrm( modrm_string );
|
|
switch ((op2 >> 3) & 0x7)
|
|
{
|
|
case 0: sprintf(s, "fadd qword ptr %s", modrm_string); break;
|
|
case 1: sprintf(s, "fmul qword ptr %s", modrm_string); break;
|
|
case 2: sprintf(s, "fcom qword ptr %s", modrm_string); break;
|
|
case 3: sprintf(s, "fcomp qword ptr %s", modrm_string); break;
|
|
case 4: sprintf(s, "fsub qword ptr %s", modrm_string); break;
|
|
case 5: sprintf(s, "fsubr qword ptr %s", modrm_string); break;
|
|
case 6: sprintf(s, "fdiv qword ptr %s", modrm_string); break;
|
|
case 7: sprintf(s, "fdivr qword ptr %s", modrm_string); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch (op2 & 0x3f)
|
|
{
|
|
case 0x00: case 0x01: case 0x02: case 0x03: case 0x04: case 0x05: case 0x06: case 0x07:
|
|
sprintf(s, "fadd st(%d),st(0)", op2 & 0x7); break;
|
|
|
|
case 0x08: case 0x09: case 0x0a: case 0x0b: case 0x0c: case 0x0d: case 0x0e: case 0x0f:
|
|
sprintf(s, "fmul st(%d),st(0)", op2 & 0x7); break;
|
|
|
|
case 0x20: case 0x21: case 0x22: case 0x23: case 0x24: case 0x25: case 0x26: case 0x27:
|
|
sprintf(s, "fsubr st(%d),st(0)", op2 & 0x7); break;
|
|
|
|
case 0x28: case 0x29: case 0x2a: case 0x2b: case 0x2c: case 0x2d: case 0x2e: case 0x2f:
|
|
sprintf(s, "fsub st(%d),st(0)", op2 & 0x7); break;
|
|
|
|
case 0x30: case 0x31: case 0x32: case 0x33: case 0x34: case 0x35: case 0x36: case 0x37:
|
|
sprintf(s, "fdivr st(%d),st(0)", op2 & 0x7); break;
|
|
|
|
case 0x38: case 0x39: case 0x3a: case 0x3b: case 0x3c: case 0x3d: case 0x3e: case 0x3f:
|
|
sprintf(s, "fdiv st(%d),st(0)", op2 & 0x7); break;
|
|
|
|
default: sprintf(s, "??? (FPU)"); break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 5: // Group DD
|
|
{
|
|
if (op2 < 0xc0)
|
|
{
|
|
pc--; // adjust fetch pointer, so modrm byte read again
|
|
opcode_ptr--;
|
|
handle_modrm( modrm_string );
|
|
switch ((op2 >> 3) & 0x7)
|
|
{
|
|
case 0: sprintf(s, "fld qword ptr %s", modrm_string); break;
|
|
case 1: sprintf(s, "fisttp qword ptr %s", modrm_string); break;
|
|
case 2: sprintf(s, "fst qword ptr %s", modrm_string); break;
|
|
case 3: sprintf(s, "fstp qword ptr %s", modrm_string); break;
|
|
case 4: sprintf(s, "frstor %s", modrm_string); break;
|
|
case 5: sprintf(s, "??? (FPU)"); break;
|
|
case 6: sprintf(s, "fsave %s", modrm_string); break;
|
|
case 7: sprintf(s, "fstsw word ptr %s", modrm_string); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch (op2 & 0x3f)
|
|
{
|
|
case 0x00: case 0x01: case 0x02: case 0x03: case 0x04: case 0x05: case 0x06: case 0x07:
|
|
sprintf(s, "ffree st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x10: case 0x11: case 0x12: case 0x13: case 0x14: case 0x15: case 0x16: case 0x17:
|
|
sprintf(s, "fst st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x18: case 0x19: case 0x1a: case 0x1b: case 0x1c: case 0x1d: case 0x1e: case 0x1f:
|
|
sprintf(s, "fstp st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x20: case 0x21: case 0x22: case 0x23: case 0x24: case 0x25: case 0x26: case 0x27:
|
|
sprintf(s, "fucom st(%d), st(0)", op2 & 0x7); break;
|
|
|
|
case 0x28: case 0x29: case 0x2a: case 0x2b: case 0x2c: case 0x2d: case 0x2e: case 0x2f:
|
|
sprintf(s, "fucomp st(%d)", op2 & 0x7); break;
|
|
|
|
default: sprintf(s, "??? (FPU)"); break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 6: // Group DE
|
|
{
|
|
if (op2 < 0xc0)
|
|
{
|
|
pc--; // adjust fetch pointer, so modrm byte read again
|
|
opcode_ptr--;
|
|
handle_modrm( modrm_string );
|
|
switch ((op2 >> 3) & 0x7)
|
|
{
|
|
case 0: sprintf(s, "fiadd word ptr %s", modrm_string); break;
|
|
case 1: sprintf(s, "fimul word ptr %s", modrm_string); break;
|
|
case 2: sprintf(s, "ficom word ptr %s", modrm_string); break;
|
|
case 3: sprintf(s, "ficomp word ptr %s", modrm_string); break;
|
|
case 4: sprintf(s, "fisub word ptr %s", modrm_string); break;
|
|
case 5: sprintf(s, "fisubr word ptr %s", modrm_string); break;
|
|
case 6: sprintf(s, "fidiv word ptr %s", modrm_string); break;
|
|
case 7: sprintf(s, "fidivr word ptr %s", modrm_string); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch (op2 & 0x3f)
|
|
{
|
|
case 0x00: case 0x01: case 0x02: case 0x03: case 0x04: case 0x05: case 0x06: case 0x07:
|
|
sprintf(s, "faddp st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x08: case 0x09: case 0x0a: case 0x0b: case 0x0c: case 0x0d: case 0x0e: case 0x0f:
|
|
sprintf(s, "fmulp st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x19: sprintf(s, "fcompp"); break;
|
|
|
|
case 0x20: case 0x21: case 0x22: case 0x23: case 0x24: case 0x25: case 0x26: case 0x27:
|
|
sprintf(s, "fsubrp st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x28: case 0x29: case 0x2a: case 0x2b: case 0x2c: case 0x2d: case 0x2e: case 0x2f:
|
|
sprintf(s, "fsubp st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x30: case 0x31: case 0x32: case 0x33: case 0x34: case 0x35: case 0x36: case 0x37:
|
|
sprintf(s, "fdivrp st(%d), st(0)", op2 & 0x7); break;
|
|
|
|
case 0x38: case 0x39: case 0x3a: case 0x3b: case 0x3c: case 0x3d: case 0x3e: case 0x3f:
|
|
sprintf(s, "fdivp st(%d)", op2 & 0x7); break;
|
|
|
|
default: sprintf(s, "??? (FPU)"); break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 7: // Group DF
|
|
{
|
|
if (op2 < 0xc0)
|
|
{
|
|
pc--; // adjust fetch pointer, so modrm byte read again
|
|
opcode_ptr--;
|
|
handle_modrm( modrm_string );
|
|
switch ((op2 >> 3) & 0x7)
|
|
{
|
|
case 0: sprintf(s, "fild word ptr %s", modrm_string); break;
|
|
case 1: sprintf(s, "fisttp word ptr %s", modrm_string); break;
|
|
case 2: sprintf(s, "fist word ptr %s", modrm_string); break;
|
|
case 3: sprintf(s, "fistp word ptr %s", modrm_string); break;
|
|
case 4: sprintf(s, "fbld %s", modrm_string); break;
|
|
case 5: sprintf(s, "fild qword ptr %s", modrm_string); break;
|
|
case 6: sprintf(s, "fbstp %s", modrm_string); break;
|
|
case 7: sprintf(s, "fistp qword ptr %s", modrm_string); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch (op2 & 0x3f)
|
|
{
|
|
case 0x20: sprintf(s, "fstsw ax"); break;
|
|
|
|
case 0x28: case 0x29: case 0x2a: case 0x2b: case 0x2c: case 0x2d: case 0x2e: case 0x2f:
|
|
sprintf(s, "fucomip st(%d)", op2 & 0x7); break;
|
|
|
|
case 0x30: case 0x31: case 0x32: case 0x33: case 0x34: case 0x35: case 0x36: case 0x37:
|
|
sprintf(s, "fcomip st(%d),st(0)", op2 & 0x7); break;
|
|
|
|
default: sprintf(s, "??? (FPU)"); break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void decode_opcode(char *s, const I386_OPCODE *op, UINT8 op1)
|
|
{
|
|
int i;
|
|
UINT8 op2;
|
|
|
|
if (op->flags & SPECIAL64)
|
|
op = &x64_opcode_alt[op->flags >> 24];
|
|
|
|
switch( op->flags & FLAGS_MASK )
|
|
{
|
|
case ISREX:
|
|
if (curmode == 64)
|
|
{
|
|
rex = op1;
|
|
operand_size = (op1 & 8) ? 2 : 1;
|
|
regex = (op1 << 1) & 8;
|
|
sibex = (op1 << 2) & 8;
|
|
rmex = (op1 << 3) & 8;
|
|
op2 = FETCH();
|
|
decode_opcode( s, &i386_opcode_table1[op2], op1 );
|
|
return;
|
|
}
|
|
break;
|
|
|
|
case OP_SIZE:
|
|
rex = regex = sibex = rmex = 0;
|
|
if (operand_size < 2)
|
|
operand_size ^= 1;
|
|
op2 = FETCH();
|
|
decode_opcode( s, &i386_opcode_table1[op2], op2 );
|
|
return;
|
|
|
|
case ADDR_SIZE:
|
|
rex = regex = sibex = rmex = 0;
|
|
if (curmode != 64)
|
|
address_size ^= 1;
|
|
else
|
|
address_size ^= 3;
|
|
op2 = FETCH();
|
|
decode_opcode( s, &i386_opcode_table1[op2], op2 );
|
|
return;
|
|
|
|
case TWO_BYTE:
|
|
if (&opcode_ptr[-2] >= opcode_ptr_base)
|
|
pre0f = opcode_ptr[-2];
|
|
op2 = FETCHD();
|
|
decode_opcode( s, &i386_opcode_table2[op2], op1 );
|
|
return;
|
|
|
|
case SEG_CS:
|
|
case SEG_DS:
|
|
case SEG_ES:
|
|
case SEG_FS:
|
|
case SEG_GS:
|
|
case SEG_SS:
|
|
rex = regex = sibex = rmex = 0;
|
|
segment = op->flags;
|
|
op2 = FETCH();
|
|
decode_opcode( s, &i386_opcode_table1[op2], op2 );
|
|
return;
|
|
|
|
case PREFIX:
|
|
op2 = FETCH();
|
|
if (op2 != 0x0f)
|
|
s += sprintf( s, "%-7s ", op->mnemonic );
|
|
decode_opcode( s, &i386_opcode_table1[op2], op2 );
|
|
return;
|
|
|
|
case GROUP:
|
|
handle_modrm( modrm_string );
|
|
for( i=0; i < ARRAY_LENGTH(group_op_table); i++ ) {
|
|
if( strcmp(op->mnemonic, group_op_table[i].mnemonic) == 0 ) {
|
|
decode_opcode( s, &group_op_table[i].opcode[MODRM_REG1], op1 );
|
|
return;
|
|
}
|
|
}
|
|
goto handle_unknown;
|
|
|
|
case FPU:
|
|
op2 = FETCHD();
|
|
handle_fpu( s, op1, op2);
|
|
return;
|
|
|
|
case MODRM:
|
|
handle_modrm( modrm_string );
|
|
break;
|
|
}
|
|
|
|
if ((op->flags & ALWAYS64) && curmode == 64)
|
|
operand_size = 2;
|
|
|
|
if ((op->flags & VAR_NAME) && operand_size > 0)
|
|
{
|
|
const char *mnemonic = op->mnemonic + strlen(op->mnemonic) + 1;
|
|
if (operand_size == 2)
|
|
mnemonic += strlen(mnemonic) + 1;
|
|
s += sprintf( s, "%-7s ", mnemonic );
|
|
}
|
|
else if (op->flags & VAR_NAME4)
|
|
{
|
|
const char *mnemonic = op->mnemonic;
|
|
int which = (pre0f == 0xf3) ? 3 : (pre0f == 0xf2) ? 2 : (pre0f == 0x66) ? 1 : 0;
|
|
while (which--)
|
|
mnemonic += strlen(mnemonic) + 1;
|
|
s += sprintf( s, "%-7s ", mnemonic );
|
|
}
|
|
else
|
|
s += sprintf( s, "%-7s ", op->mnemonic );
|
|
dasm_flags = op->dasm_flags;
|
|
|
|
if( op->param1 != 0 ) {
|
|
s = handle_param( s, op->param1 );
|
|
}
|
|
|
|
if( op->param2 != 0 ) {
|
|
s += sprintf( s, "," );
|
|
s = handle_param( s, op->param2 );
|
|
}
|
|
|
|
if( op->param3 != 0 ) {
|
|
s += sprintf( s, "," );
|
|
s = handle_param( s, op->param3 );
|
|
}
|
|
return;
|
|
|
|
handle_unknown:
|
|
sprintf(s, "???");
|
|
}
|
|
|
|
int i386_dasm_one_ex(char *buffer, UINT64 eip, const UINT8 *oprom, int mode)
|
|
{
|
|
UINT8 op;
|
|
|
|
opcode_ptr = opcode_ptr_base = oprom;
|
|
address_size = (mode == 16) ? 0 : (mode == 32) ? 1 : 2;
|
|
operand_size = (mode == 16) ? 0 : 1;
|
|
pc = eip;
|
|
dasm_flags = 0;
|
|
segment = 0;
|
|
curmode = mode;
|
|
pre0f = 0;
|
|
rex = regex = sibex = rmex = 0;
|
|
|
|
op = FETCH();
|
|
|
|
decode_opcode( buffer, &i386_opcode_table1[op], op );
|
|
return (pc-eip) | dasm_flags | DASMFLAG_SUPPORTED;
|
|
}
|
|
|
|
int i386_dasm_one(char *buffer, offs_t eip, const UINT8 *oprom, int mode)
|
|
{
|
|
return i386_dasm_one_ex(buffer, eip, oprom, mode);
|
|
}
|
|
|
|
CPU_DISASSEMBLE( x86_16 )
|
|
{
|
|
return i386_dasm_one_ex(buffer, pc, oprom, 16);
|
|
}
|
|
|
|
CPU_DISASSEMBLE( x86_32 )
|
|
{
|
|
return i386_dasm_one_ex(buffer, pc, oprom, 32);
|
|
}
|
|
|
|
CPU_DISASSEMBLE( x86_64 )
|
|
{
|
|
return i386_dasm_one_ex(buffer, pc, oprom, 64);
|
|
}
|