mupen64plus-oldsvn/r4300/x86/assemble.c

1297 lines
22 KiB
C

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* Mupen64plus - assemble.c *
* Mupen64Plus homepage: http://code.google.com/p/mupen64plus/ *
* Copyright (C) 2002 Hacktarux *
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
#include <stdlib.h>
#include <stdio.h>
#include "assemble.h"
#include "../recomph.h"
#include "../r4300.h"
typedef struct _jump_table
{
unsigned int mi_addr;
unsigned int pc_addr;
} jump_table;
static jump_table *jumps_table = NULL;
static int jumps_number, max_jumps_number;
void init_assembler(void *block_jumps_table, int block_jumps_number, void *block_riprel_table, int block_riprel_number)
{
if (block_jumps_table)
{
jumps_table = block_jumps_table;
jumps_number = block_jumps_number;
max_jumps_number = jumps_number;
}
else
{
jumps_table = malloc(1000*sizeof(jump_table));
jumps_number = 0;
max_jumps_number = 1000;
}
}
void free_assembler(void **block_jumps_table, int *block_jumps_number, void **block_riprel_table, int *block_riprel_number)
{
*block_jumps_table = jumps_table;
*block_jumps_number = jumps_number;
*block_riprel_table = NULL; /* RIP-relative addressing is only for x86-64 */
*block_riprel_number = 0;
}
static void add_jump(unsigned int pc_addr, unsigned int mi_addr)
{
if (jumps_number == max_jumps_number)
{
max_jumps_number += 1000;
jumps_table = realloc(jumps_table, max_jumps_number*sizeof(jump_table));
}
jumps_table[jumps_number].pc_addr = pc_addr;
jumps_table[jumps_number].mi_addr = mi_addr;
jumps_number++;
}
static void put8(unsigned char octet)
{
(*inst_pointer)[code_length] = octet;
code_length++;
if (code_length == max_code_length)
{
*inst_pointer = realloc_exec(*inst_pointer, max_code_length, max_code_length+8192);
max_code_length += 8192;
}
}
static void put16(unsigned short word)
{
if ((code_length+2) >= max_code_length)
{
*inst_pointer = realloc_exec(*inst_pointer, max_code_length, max_code_length+8192);
max_code_length += 8192;
}
*((unsigned short *)(&(*inst_pointer)[code_length])) = word;
code_length+=2;
}
static void put32(unsigned int dword)
{
if ((code_length+4) >= max_code_length)
{
*inst_pointer = realloc_exec(*inst_pointer, max_code_length, max_code_length+8192);
max_code_length += 8192;
}
*((unsigned int *)(&(*inst_pointer)[code_length])) = dword;
code_length+=4;
}
void passe2(precomp_instr *dest, int start, int end, precomp_block *block)
{
unsigned int i, real_code_length, addr_dest;
build_wrappers(dest, start, end, block);
real_code_length = code_length;
for (i=0; i<jumps_number; i++)
{
code_length = jumps_table[i].pc_addr;
if (dest[(jumps_table[i].mi_addr - dest[0].addr)/4].reg_cache_infos.need_map)
{
addr_dest = (unsigned int)dest[(jumps_table[i].mi_addr - dest[0].addr)/4].reg_cache_infos.jump_wrapper;
put32(addr_dest-((unsigned int)block->code+code_length)-4);
}
else
{
addr_dest = dest[(jumps_table[i].mi_addr - dest[0].addr)/4].local_addr;
put32(addr_dest-code_length-4);
}
}
code_length = real_code_length;
}
static unsigned int g_jump_start8 = 0;
static unsigned int g_jump_start32 = 0;
void jump_start_rel8(void)
{
g_jump_start8 = code_length;
}
void jump_start_rel32(void)
{
g_jump_start32 = code_length;
}
void jump_end_rel8(void)
{
unsigned int jump_end = code_length;
int jump_vec = jump_end - g_jump_start8;
if (jump_vec > 127 || jump_vec < -128)
{
printf("Error: 8-bit relative jump too long! From %x to %x\n", g_jump_start8, jump_end);
asm(" int $3; ");
}
code_length = g_jump_start8 - 1;
put8(jump_vec);
code_length = jump_end;
}
void jump_end_rel32(void)
{
unsigned int jump_end = code_length;
int jump_vec = jump_end - g_jump_start32;
code_length = g_jump_start32 - 4;
put32(jump_vec);
code_length = jump_end;
}
void push_reg32(int reg32)
{
put8(0x50+reg32);
}
void pop_reg32(int reg32)
{
put8(0x58+reg32);
}
void mov_eax_memoffs32(unsigned int *memoffs32)
{
put8(0xA1);
put32((unsigned int)(memoffs32));
}
void mov_memoffs32_eax(unsigned int *memoffs32)
{
put8(0xA3);
put32((unsigned int)(memoffs32));
}
void mov_ax_memoffs16(unsigned short *memoffs16)
{
put8(0x66);
put8(0xA1);
put32((unsigned int)(memoffs16));
}
void mov_memoffs16_ax(unsigned short *memoffs16)
{
put8(0x66);
put8(0xA3);
put32((unsigned int)(memoffs16));
}
void mov_al_memoffs8(unsigned char *memoffs8)
{
put8(0xA0);
put32((unsigned int)(memoffs8));
}
void mov_memoffs8_al(unsigned char *memoffs8)
{
put8(0xA2);
put32((unsigned int)(memoffs8));
}
void mov_m8_imm8(unsigned char *m8, unsigned char imm8)
{
put8(0xC6);
put8(0x05);
put32((unsigned int)(m8));
put8(imm8);
}
void mov_m8_reg8(unsigned char *m8, int reg8)
{
put8(0x88);
put8((reg8 << 3) | 5);
put32((unsigned int)(m8));
}
void mov_reg16_m16(int reg16, unsigned short *m16)
{
put8(0x66);
put8(0x8B);
put8((reg16 << 3) | 5);
put32((unsigned int)(m16));
}
void mov_m16_reg16(unsigned short *m16, int reg16)
{
put8(0x66);
put8(0x89);
put8((reg16 << 3) | 5);
put32((unsigned int)(m16));
}
void cmp_reg32_m32(int reg32, unsigned int *m32)
{
put8(0x3B);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void cmp_reg32_reg32(int reg1, int reg2)
{
put8(0x39);
put8((reg2 << 3) | reg1 | 0xC0);
}
void cmp_reg32_imm8(int reg32, unsigned char imm8)
{
put8(0x83);
put8(0xF8 + reg32);
put8(imm8);
}
void cmp_preg32pimm32_imm8(int reg32, unsigned int imm32, unsigned char imm8)
{
put8(0x80);
put8(0xB8 + reg32);
put32(imm32);
put8(imm8);
}
void cmp_reg32_imm32(int reg32, unsigned int imm32)
{
put8(0x81);
put8(0xF8 + reg32);
put32(imm32);
}
void test_reg32_imm32(int reg32, unsigned int imm32)
{
put8(0xF7);
put8(0xC0 + reg32);
put32(imm32);
}
void test_m32_imm32(unsigned int *m32, unsigned int imm32)
{
put8(0xF7);
put8(0x05);
put32((unsigned int)m32);
put32(imm32);
}
void cmp_al_imm8(unsigned char imm8)
{
put8(0x3C);
put8(imm8);
}
void add_m32_reg32(unsigned int *m32, int reg32)
{
put8(0x01);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void sub_reg32_m32(int reg32, unsigned int *m32)
{
put8(0x2B);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void sub_reg32_reg32(int reg1, int reg2)
{
put8(0x29);
put8((reg2 << 3) | reg1 | 0xC0);
}
void sbb_reg32_reg32(int reg1, int reg2)
{
put8(0x19);
put8((reg2 << 3) | reg1 | 0xC0);
}
void sub_reg32_imm32(int reg32, unsigned int imm32)
{
put8(0x81);
put8(0xE8 + reg32);
put32(imm32);
}
void sub_eax_imm32(unsigned int imm32)
{
put8(0x2D);
put32(imm32);
}
void jne_rj(unsigned char saut)
{
put8(0x75);
put8(saut);
}
void je_rj(unsigned char saut)
{
put8(0x74);
put8(saut);
}
void jb_rj(unsigned char saut)
{
put8(0x72);
put8(saut);
}
void jbe_rj(unsigned char saut)
{
put8(0x76);
put8(saut);
}
void ja_rj(unsigned char saut)
{
put8(0x77);
put8(saut);
}
void jae_rj(unsigned char saut)
{
put8(0x73);
put8(saut);
}
void jle_rj(unsigned char saut)
{
put8(0x7E);
put8(saut);
}
void jge_rj(unsigned char saut)
{
put8(0x7D);
put8(saut);
}
void jg_rj(unsigned char saut)
{
put8(0x7F);
put8(saut);
}
void jl_rj(unsigned char saut)
{
put8(0x7C);
put8(saut);
}
void jp_rj(unsigned char saut)
{
put8(0x7A);
put8(saut);
}
void je_near(unsigned int mi_addr)
{
put8(0x0F);
put8(0x84);
put32(0);
add_jump(code_length-4, mi_addr);
}
void je_near_rj(unsigned int saut)
{
put8(0x0F);
put8(0x84);
put32(saut);
}
void jl_near(unsigned int mi_addr)
{
put8(0x0F);
put8(0x8C);
put32(0);
add_jump(code_length-4, mi_addr);
}
void jl_near_rj(unsigned int saut)
{
put8(0x0F);
put8(0x8C);
put32(saut);
}
void jne_near(unsigned int mi_addr)
{
put8(0x0F);
put8(0x85);
put32(0);
add_jump(code_length-4, mi_addr);
}
void jne_near_rj(unsigned int saut)
{
put8(0x0F);
put8(0x85);
put32(saut);
}
void jge_near(unsigned int mi_addr)
{
put8(0x0F);
put8(0x8D);
put32(0);
add_jump(code_length-4, mi_addr);
}
void jge_near_rj(unsigned int saut)
{
put8(0x0F);
put8(0x8D);
put32(saut);
}
void jg_near(unsigned int mi_addr)
{
put8(0x0F);
put8(0x8F);
put32(0);
add_jump(code_length-4, mi_addr);
}
void jle_near(unsigned int mi_addr)
{
put8(0x0F);
put8(0x8E);
put32(0);
add_jump(code_length-4, mi_addr);
}
void jle_near_rj(unsigned int saut)
{
put8(0x0F);
put8(0x8E);
put32(saut);
}
void mov_reg32_imm32(int reg32, unsigned int imm32)
{
put8(0xB8+reg32);
put32(imm32);
}
void jmp_imm(int saut)
{
put8(0xE9);
put32(saut);
}
void jmp_imm_short(char saut)
{
put8(0xEB);
put8(saut);
}
void dec_reg32(int reg32)
{
put8(0x48+reg32);
}
void inc_reg32(int reg32)
{
put8(0x40+reg32);
}
void or_m32_imm32(unsigned int *m32, unsigned int imm32)
{
put8(0x81);
put8(0x0D);
put32((unsigned int)(m32));
put32(imm32);
}
void or_m32_reg32(unsigned int *m32, unsigned int reg32)
{
put8(0x09);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void or_reg32_m32(unsigned int reg32, unsigned int *m32)
{
put8(0x0B);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void or_reg32_reg32(unsigned int reg1, unsigned int reg2)
{
put8(0x09);
put8(0xC0 | (reg2 << 3) | reg1);
}
void and_reg32_reg32(unsigned int reg1, unsigned int reg2)
{
put8(0x21);
put8(0xC0 | (reg2 << 3) | reg1);
}
void and_m32_imm32(unsigned int *m32, unsigned int imm32)
{
put8(0x81);
put8(0x25);
put32((unsigned int)(m32));
put32(imm32);
}
void and_reg32_m32(unsigned int reg32, unsigned int *m32)
{
put8(0x23);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void xor_reg32_reg32(unsigned int reg1, unsigned int reg2)
{
put8(0x31);
put8(0xC0 | (reg2 << 3) | reg1);
}
void xor_reg32_m32(unsigned int reg32, unsigned int *m32)
{
put8(0x33);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void add_m32_imm32(unsigned int *m32, unsigned int imm32)
{
put8(0x81);
put8(0x05);
put32((unsigned int)(m32));
put32(imm32);
}
void add_m32_imm8(unsigned int *m32, unsigned char imm8)
{
put8(0x83);
put8(0x05);
put32((unsigned int)(m32));
put8(imm8);
}
void sub_m32_imm32(unsigned int *m32, unsigned int imm32)
{
put8(0x81);
put8(0x2D);
put32((unsigned int)(m32));
put32(imm32);
}
void push_imm32(unsigned int imm32)
{
put8(0x68);
put32(imm32);
}
void add_reg32_imm8(unsigned int reg32, unsigned char imm8)
{
put8(0x83);
put8(0xC0+reg32);
put8(imm8);
}
void add_reg32_imm32(unsigned int reg32, unsigned int imm32)
{
put8(0x81);
put8(0xC0+reg32);
put32(imm32);
}
void inc_m32(unsigned int *m32)
{
put8(0xFF);
put8(0x05);
put32((unsigned int)(m32));
}
void cmp_m32_imm32(unsigned int *m32, unsigned int imm32)
{
put8(0x81);
put8(0x3D);
put32((unsigned int)(m32));
put32(imm32);
}
void cmp_m32_imm8(unsigned int *m32, unsigned char imm8)
{
put8(0x83);
put8(0x3D);
put32((unsigned int)(m32));
put8(imm8);
}
void cmp_m8_imm8(unsigned char *m8, unsigned char imm8)
{
put8(0x80);
put8(0x3D);
put32((unsigned int)(m8));
put8(imm8);
}
void cmp_eax_imm32(unsigned int imm32)
{
put8(0x3D);
put32(imm32);
}
void mov_m32_imm32(unsigned int *m32, unsigned int imm32)
{
put8(0xC7);
put8(0x05);
put32((unsigned int)(m32));
put32(imm32);
}
void jmp(unsigned int mi_addr)
{
put8(0xE9);
put32(0);
add_jump(code_length-4, mi_addr);
}
void cdq(void)
{
put8(0x99);
}
void cwde(void)
{
put8(0x98);
}
void cbw(void)
{
put8(0x66);
put8(0x98);
}
void mov_m32_reg32(unsigned int *m32, unsigned int reg32)
{
put8(0x89);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void ret(void)
{
put8(0xC3);
}
void call_reg32(unsigned int reg32)
{
put8(0xFF);
put8(0xD0+reg32);
}
void call_m32(unsigned int *m32)
{
put8(0xFF);
put8(0x15);
put32((unsigned int)(m32));
}
void shr_reg32_imm8(unsigned int reg32, unsigned char imm8)
{
put8(0xC1);
put8(0xE8+reg32);
put8(imm8);
}
void shr_reg32_cl(unsigned int reg32)
{
put8(0xD3);
put8(0xE8+reg32);
}
void sar_reg32_cl(unsigned int reg32)
{
put8(0xD3);
put8(0xF8+reg32);
}
void shl_reg32_cl(unsigned int reg32)
{
put8(0xD3);
put8(0xE0+reg32);
}
void shld_reg32_reg32_cl(unsigned int reg1, unsigned int reg2)
{
put8(0x0F);
put8(0xA5);
put8(0xC0 | (reg2 << 3) | reg1);
}
void shld_reg32_reg32_imm8(unsigned int reg1, unsigned int reg2, unsigned char imm8)
{
put8(0x0F);
put8(0xA4);
put8(0xC0 | (reg2 << 3) | reg1);
put8(imm8);
}
void shrd_reg32_reg32_cl(unsigned int reg1, unsigned int reg2)
{
put8(0x0F);
put8(0xAD);
put8(0xC0 | (reg2 << 3) | reg1);
}
void sar_reg32_imm8(unsigned int reg32, unsigned char imm8)
{
put8(0xC1);
put8(0xF8+reg32);
put8(imm8);
}
void shrd_reg32_reg32_imm8(unsigned int reg1, unsigned int reg2, unsigned char imm8)
{
put8(0x0F);
put8(0xAC);
put8(0xC0 | (reg2 << 3) | reg1);
put8(imm8);
}
void mul_m32(unsigned int *m32)
{
put8(0xF7);
put8(0x25);
put32((unsigned int)(m32));
}
void imul_m32(unsigned int *m32)
{
put8(0xF7);
put8(0x2D);
put32((unsigned int)(m32));
}
void imul_reg32(unsigned int reg32)
{
put8(0xF7);
put8(0xE8+reg32);
}
void mul_reg32(unsigned int reg32)
{
put8(0xF7);
put8(0xE0+reg32);
}
void idiv_reg32(unsigned int reg32)
{
put8(0xF7);
put8(0xF8+reg32);
}
void div_reg32(unsigned int reg32)
{
put8(0xF7);
put8(0xF0+reg32);
}
void idiv_m32(unsigned int *m32)
{
put8(0xF7);
put8(0x3D);
put32((unsigned int)(m32));
}
void div_m32(unsigned int *m32)
{
put8(0xF7);
put8(0x35);
put32((unsigned int)(m32));
}
void add_reg32_reg32(unsigned int reg1, unsigned int reg2)
{
put8(0x01);
put8(0xC0 | (reg2 << 3) | reg1);
}
void adc_reg32_reg32(unsigned int reg1, unsigned int reg2)
{
put8(0x11);
put8(0xC0 | (reg2 << 3) | reg1);
}
void add_reg32_m32(unsigned int reg32, unsigned int *m32)
{
put8(0x03);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void adc_reg32_m32(unsigned int reg32, unsigned int *m32)
{
put8(0x13);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void adc_reg32_imm32(unsigned int reg32, unsigned int imm32)
{
put8(0x81);
put8(0xD0 + reg32);
put32(imm32);
}
void jmp_reg32(unsigned int reg32)
{
put8(0xFF);
put8(0xE0 + reg32);
}
void jmp_m32(unsigned int *m32)
{
put8(0xFF);
put8(0x25);
put32((unsigned int)(m32));
}
void mov_reg32_preg32(unsigned int reg1, unsigned int reg2)
{
put8(0x8B);
put8((reg1 << 3) | reg2);
}
void mov_preg32_reg32(int reg1, int reg2)
{
put8(0x89);
put8((reg2 << 3) | reg1);
}
void mov_reg32_preg32preg32pimm32(int reg1, int reg2, int reg3, unsigned int imm32)
{
put8(0x8B);
put8((reg1 << 3) | 0x84);
put8(reg2 | (reg3 << 3));
put32(imm32);
}
void mov_reg32_preg32pimm32(int reg1, int reg2, unsigned int imm32)
{
put8(0x8B);
put8(0x80 | (reg1 << 3) | reg2);
put32(imm32);
}
void mov_reg32_preg32x4preg32(int reg1, int reg2, int reg3)
{
put8(0x8B);
put8((reg1 << 3) | 4);
put8(0x80 | (reg2 << 3) | reg3);
}
void mov_reg32_preg32x4preg32pimm32(int reg1, int reg2, int reg3, unsigned int imm32)
{
put8(0x8B);
put8((reg1 << 3) | 0x84);
put8(0x80 | (reg2 << 3) | reg3);
put32(imm32);
}
void mov_reg32_preg32x4pimm32(int reg1, int reg2, unsigned int imm32)
{
put8(0x8B);
put8((reg1 << 3) | 4);
put8(0x80 | (reg2 << 3) | 5);
put32(imm32);
}
void mov_preg32preg32pimm32_reg8(int reg1, int reg2, unsigned int imm32, int reg8)
{
put8(0x88);
put8(0x84 | (reg8 << 3));
put8((reg2 << 3) | reg1);
put32(imm32);
}
void mov_preg32pimm32_reg8(int reg32, unsigned int imm32, int reg8)
{
put8(0x88);
put8(0x80 | reg32 | (reg8 << 3));
put32(imm32);
}
void mov_preg32pimm32_imm8(int reg32, unsigned int imm32, unsigned char imm8)
{
put8(0xC6);
put8(0x80 + reg32);
put32(imm32);
put8(imm8);
}
void mov_preg32pimm32_reg16(int reg32, unsigned int imm32, int reg16)
{
put8(0x66);
put8(0x89);
put8(0x80 | reg32 | (reg16 << 3));
put32(imm32);
}
void mov_preg32pimm32_reg32(int reg1, unsigned int imm32, int reg2)
{
put8(0x89);
put8(0x80 | reg1 | (reg2 << 3));
put32(imm32);
}
void add_eax_imm32(unsigned int imm32)
{
put8(0x05);
put32(imm32);
}
void shl_reg32_imm8(unsigned int reg32, unsigned char imm8)
{
put8(0xC1);
put8(0xE0 + reg32);
put8(imm8);
}
void mov_reg32_m32(unsigned int reg32, unsigned int* m32)
{
put8(0x8B);
put8((reg32 << 3) | 5);
put32((unsigned int)(m32));
}
void mov_reg8_m8(int reg8, unsigned char *m8)
{
put8(0x8A);
put8((reg8 << 3) | 5);
put32((unsigned int)(m8));
}
void and_eax_imm32(unsigned int imm32)
{
put8(0x25);
put32(imm32);
}
void and_reg32_imm32(int reg32, unsigned int imm32)
{
put8(0x81);
put8(0xE0 + reg32);
put32(imm32);
}
void or_reg32_imm32(int reg32, unsigned int imm32)
{
put8(0x81);
put8(0xC8 + reg32);
put32(imm32);
}
void and_reg32_imm8(int reg32, unsigned char imm8)
{
put8(0x83);
put8(0xE0 + reg32);
put8(imm8);
}
void and_ax_imm16(unsigned short imm16)
{
put8(0x66);
put8(0x25);
put16(imm16);
}
void and_al_imm8(unsigned char imm8)
{
put8(0x24);
put8(imm8);
}
void or_ax_imm16(unsigned short imm16)
{
put8(0x66);
put8(0x0D);
put16(imm16);
}
void or_eax_imm32(unsigned int imm32)
{
put8(0x0D);
put32(imm32);
}
void xor_ax_imm16(unsigned short imm16)
{
put8(0x66);
put8(0x35);
put16(imm16);
}
void xor_al_imm8(unsigned char imm8)
{
put8(0x34);
put8(imm8);
}
void xor_reg32_imm32(int reg32, unsigned int imm32)
{
put8(0x81);
put8(0xF0 + reg32);
put32(imm32);
}
void xor_reg8_imm8(int reg8, unsigned char imm8)
{
put8(0x80);
put8(0xF0 + reg8);
put8(imm8);
}
void nop(void)
{
put8(0x90);
}
void mov_reg32_reg32(unsigned int reg1, unsigned int reg2)
{
if (reg1 == reg2) return;
put8(0x89);
put8(0xC0 | (reg2 << 3) | reg1);
}
void not_reg32(unsigned int reg32)
{
put8(0xF7);
put8(0xD0 + reg32);
}
void movsx_reg32_m8(int reg32, unsigned char *m8)
{
put8(0x0F);
put8(0xBE);
put8((reg32 << 3) | 5);
put32((unsigned int)(m8));
}
void movsx_reg32_reg8(int reg32, int reg8)
{
put8(0x0F);
put8(0xBE);
put8((reg32 << 3) | reg8 | 0xC0);
}
void movsx_reg32_8preg32pimm32(int reg1, int reg2, unsigned int imm32)
{
put8(0x0F);
put8(0xBE);
put8((reg1 << 3) | reg2 | 0x80);
put32(imm32);
}
void movsx_reg32_16preg32pimm32(int reg1, int reg2, unsigned int imm32)
{
put8(0x0F);
put8(0xBF);
put8((reg1 << 3) | reg2 | 0x80);
put32(imm32);
}
void movsx_reg32_reg16(int reg32, int reg16)
{
put8(0x0F);
put8(0xBF);
put8((reg32 << 3) | reg16 | 0xC0);
}
void movsx_reg32_m16(int reg32, unsigned short *m16)
{
put8(0x0F);
put8(0xBF);
put8((reg32 << 3) | 5);
put32((unsigned int)(m16));
}
void fldcw_m16(unsigned short *m16)
{
put8(0xD9);
put8(0x2D);
put32((unsigned int)(m16));
}
void fld_preg32_dword(int reg32)
{
put8(0xD9);
put8(reg32);
}
void fdiv_preg32_dword(int reg32)
{
put8(0xD8);
put8(0x30 + reg32);
}
void fstp_preg32_dword(int reg32)
{
put8(0xD9);
put8(0x18 + reg32);
}
void fchs(void)
{
put8(0xD9);
put8(0xE0);
}
void fstp_preg32_qword(int reg32)
{
put8(0xDD);
put8(0x18 + reg32);
}
void fadd_preg32_dword(int reg32)
{
put8(0xD8);
put8(reg32);
}
void fsub_preg32_dword(int reg32)
{
put8(0xD8);
put8(0x20 + reg32);
}
void fmul_preg32_dword(int reg32)
{
put8(0xD8);
put8(0x08 + reg32);
}
void fcomp_preg32_dword(int reg32)
{
put8(0xD8);
put8(0x18 + reg32);
}
void fistp_preg32_dword(int reg32)
{
put8(0xDB);
put8(0x18 + reg32);
}
void fistp_m32(unsigned int *m32)
{
put8(0xDB);
put8(0x1D);
put32((unsigned int)(m32));
}
void fistp_preg32_qword(int reg32)
{
put8(0xDF);
put8(0x38 + reg32);
}
void fistp_m64(unsigned long long *m64)
{
put8(0xDF);
put8(0x3D);
put32((unsigned int)(m64));
}
void fld_preg32_qword(int reg32)
{
put8(0xDD);
put8(reg32);
}
void fild_preg32_qword(int reg32)
{
put8(0xDF);
put8(0x28+reg32);
}
void fild_preg32_dword(int reg32)
{
put8(0xDB);
put8(reg32);
}
void fadd_preg32_qword(int reg32)
{
put8(0xDC);
put8(reg32);
}
void fdiv_preg32_qword(int reg32)
{
put8(0xDC);
put8(0x30 + reg32);
}
void fsub_preg32_qword(int reg32)
{
put8(0xDC);
put8(0x20 + reg32);
}
void fmul_preg32_qword(int reg32)
{
put8(0xDC);
put8(0x08 + reg32);
}
void fsqrt(void)
{
put8(0xD9);
put8(0xFA);
}
void fabs_(void)
{
put8(0xD9);
put8(0xE1);
}
void fcomip_fpreg(int fpreg)
{
put8(0xDF);
put8(0xF0 + fpreg);
}
void fucomip_fpreg(int fpreg)
{
put8(0xDF);
put8(0xE8 + fpreg);
}
void ffree_fpreg(int fpreg)
{
put8(0xDD);
put8(0xC0 + fpreg);
}