Implement basic encodings for xchg

This commit is contained in:
Maxime Chevalier-Boisvert 2021-05-12 10:18:46 -04:00 коммит произвёл Alan Wu
Родитель 5c2f74fc32
Коммит 6b5d26dc78
3 изменённых файлов: 30 добавлений и 1 удалений

Просмотреть файл

@ -1682,6 +1682,29 @@ void ud2(codeblock_t* cb)
cb_write_bytes(cb, 2, 0x0F, 0x0B);
}
/// xchg - Exchange Register/Memory with Register
void xchg(codeblock_t* cb, x86opnd_t rm_opnd, x86opnd_t r_opnd)
{
assert (rm_opnd.num_bits == 64);
assert (r_opnd.num_bits == 64);
assert (rm_opnd.type == OPND_REG);
assert (r_opnd.type == OPND_REG);
// If we're exchanging with RAX
if (rm_opnd.type == OPND_REG && rm_opnd.as.reg.reg_no == RAX.as.reg.reg_no)
{
// Write the REX byte
cb_write_rex(cb, rm_opnd.num_bits == 64, 0, 0, r_opnd.as.reg.reg_no);
// Write the opcode and register number
cb_write_byte(cb, 0x90 + (r_opnd.as.reg.reg_no & 7));
}
else
{
cb_write_rm(cb, rm_opnd.num_bits == 16, rm_opnd.num_bits == 64, r_opnd, rm_opnd, 0xFF, 1, 0x87);
}
}
/// xor - Exclusive bitwise OR
void xor(codeblock_t* cb, x86opnd_t opnd0, x86opnd_t opnd1)
{

Просмотреть файл

@ -377,8 +377,8 @@ void shr(codeblock_t* cb, x86opnd_t opnd0, x86opnd_t opnd1);
void sub(codeblock_t* cb, x86opnd_t opnd0, x86opnd_t opnd1);
void test(codeblock_t* cb, x86opnd_t rm_opnd, x86opnd_t test_opnd);
void ud2(codeblock_t* cb);
void xchg(codeblock_t* cb, x86opnd_t rm_opnd, x86opnd_t r_opnd);
void xor(codeblock_t* cb, x86opnd_t opnd0, x86opnd_t opnd1);
void cb_write_lock_prefix(codeblock_t* cb);
#endif

Просмотреть файл

@ -341,6 +341,12 @@ void run_tests()
cb_set_pos(cb, 0); test(cb, RAX, RSI); check_bytes(cb, "4885F0");
cb_set_pos(cb, 0); test(cb, mem_opnd(64, RSI, 64), imm_opnd(~0x08)); check_bytes(cb, "48F74640F7FFFFFF");
// xchg
cb_set_pos(cb, 0); xchg(cb, RAX, RCX); check_bytes(cb, "4891");
cb_set_pos(cb, 0); xchg(cb, RAX, R13); check_bytes(cb, "4995");
cb_set_pos(cb, 0); xchg(cb, RCX, RBX); check_bytes(cb, "4887D9");
cb_set_pos(cb, 0); xchg(cb, R9, R15); check_bytes(cb, "4D87F9");
// xor
cb_set_pos(cb, 0); xor(cb, EAX, EAX); check_bytes(cb, "31C0");