#ifndef _CODEGEN_OPS_HELPERS_H_ #define _CODEGEN_OPS_HELPERS_H_ #include "386_common.h" #include "codegen_backend.h" static inline int LOAD_SP_WITH_OFFSET(ir_data_t *ir, int offset) { if (stack32) { if (offset) { uop_ADD_IMM(ir, IREG_eaaddr, IREG_ESP, offset); return IREG_eaaddr; } else return IREG_ESP; } else { if (offset) { uop_ADD_IMM(ir, IREG_eaaddr_W, IREG_SP, offset); uop_MOVZX(ir, IREG_eaaddr, IREG_eaaddr_W); return IREG_eaaddr; } else { uop_MOVZX(ir, IREG_eaaddr, IREG_SP); return IREG_eaaddr; } } } static inline int LOAD_SP(ir_data_t *ir) { return LOAD_SP_WITH_OFFSET(ir, 0); } static inline void ADD_SP(ir_data_t *ir, int offset) { if (stack32) uop_ADD_IMM(ir, IREG_ESP, IREG_ESP, offset); else uop_ADD_IMM(ir, IREG_SP, IREG_SP, offset); } static inline void SUB_SP(ir_data_t *ir, int offset) { if (stack32) uop_SUB_IMM(ir, IREG_ESP, IREG_ESP, offset); else uop_SUB_IMM(ir, IREG_SP, IREG_SP, offset); } static inline void fpu_POP(codeblock_t *block, ir_data_t *ir) { if (block->flags & CODEBLOCK_STATIC_TOP) uop_MOV_IMM(ir, IREG_FPU_TOP, cpu_state.TOP + 1); else uop_ADD_IMM(ir, IREG_FPU_TOP, IREG_FPU_TOP, 1); } static inline void fpu_POP2(codeblock_t *block, ir_data_t *ir) { if (block->flags & CODEBLOCK_STATIC_TOP) uop_MOV_IMM(ir, IREG_FPU_TOP, cpu_state.TOP + 2); else uop_ADD_IMM(ir, IREG_FPU_TOP, IREG_FPU_TOP, 2); } static inline void fpu_PUSH(codeblock_t *block, ir_data_t *ir) { if (block->flags & CODEBLOCK_STATIC_TOP) uop_MOV_IMM(ir, IREG_FPU_TOP, cpu_state.TOP - 1); else uop_SUB_IMM(ir, IREG_FPU_TOP, IREG_FPU_TOP, 1); } static inline void CHECK_SEG_LIMITS(codeblock_t *block, ir_data_t *ir, x86seg *seg, int addr_reg, int end_offset) { if ((seg == &cpu_state.seg_ds && codegen_flat_ds && !(cpu_cur_status & CPU_STATUS_NOTFLATDS)) || (seg == &cpu_state.seg_ss && codegen_flat_ss && !(cpu_cur_status & CPU_STATUS_NOTFLATSS))) return; uop_CMP_JB(ir, addr_reg, ireg_seg_limit_low(seg), codegen_gpf_rout); if (end_offset) { uop_ADD_IMM(ir, IREG_temp3, addr_reg, end_offset); uop_CMP_JNBE(ir, IREG_temp3, ireg_seg_limit_high(seg), codegen_gpf_rout); } else uop_CMP_JNBE(ir, addr_reg, ireg_seg_limit_high(seg), codegen_gpf_rout); } static inline void LOAD_IMMEDIATE_FROM_RAM_8(codeblock_t *block, ir_data_t *ir, int dest_reg, uint32_t addr) { uop_MOVZX_REG_PTR_8(ir, dest_reg, get_ram_ptr(addr)); } void LOAD_IMMEDIATE_FROM_RAM_16_unaligned(codeblock_t *block, ir_data_t *ir, int dest_reg, uint32_t addr); static inline void LOAD_IMMEDIATE_FROM_RAM_16(codeblock_t *block, ir_data_t *ir, int dest_reg, uint32_t addr) { if ((addr & 0xfff) == 0xfff) LOAD_IMMEDIATE_FROM_RAM_16_unaligned(block, ir, dest_reg, addr); else uop_MOVZX_REG_PTR_16(ir, dest_reg, get_ram_ptr(addr)); } void LOAD_IMMEDIATE_FROM_RAM_32_unaligned(codeblock_t *block, ir_data_t *ir, int dest_reg, uint32_t addr); static inline void LOAD_IMMEDIATE_FROM_RAM_32(codeblock_t *block, ir_data_t *ir, int dest_reg, uint32_t addr) { if ((addr & 0xfff) >= 0xffd) LOAD_IMMEDIATE_FROM_RAM_32_unaligned(block, ir, dest_reg, addr); else uop_MOV_REG_PTR(ir, dest_reg, get_ram_ptr(addr)); } int codegen_can_unroll_full(codeblock_t *block, ir_data_t *ir, uint32_t next_pc, uint32_t dest_addr); static inline int codegen_can_unroll(codeblock_t *block, ir_data_t *ir, uint32_t next_pc, uint32_t dest_addr) { if (block->flags & CODEBLOCK_BYTE_MASK) return 0; /*Is dest within block?*/ if (dest_addr > next_pc) return 0; if ((cs + dest_addr) < block->pc) return 0; return codegen_can_unroll_full(block, ir, next_pc, dest_addr); } #endif /* _CODEGEN_OPS_HELPERS_H_ */