42 #define OPCODE_NOP (0x90) 43 #define OPCODE_PUSH_R64 (0x50) 44 #define OPCODE_PUSH_I64 (0x68) 45 #define OPCODE_PUSH_M64 (0xff) 46 #define OPCODE_POP_R64 (0x58) 47 #define OPCODE_RET (0xc3) 48 #define OPCODE_MOV_I8_TO_R8 (0xb0) 49 #define OPCODE_MOV_I64_TO_R64 (0xb8) 50 #define OPCODE_MOV_I32_TO_RM32 (0xc7) 51 #define OPCODE_MOV_R8_TO_RM8 (0x88) 52 #define OPCODE_MOV_R64_TO_RM64 (0x89) 53 #define OPCODE_MOV_RM64_TO_R64 (0x8b) 54 #define OPCODE_MOVZX_RM8_TO_R64 (0xb6) 55 #define OPCODE_MOVZX_RM16_TO_R64 (0xb7) 56 #define OPCODE_LEA_MEM_TO_R64 (0x8d) 57 #define OPCODE_AND_R64_TO_RM64 (0x21) 58 #define OPCODE_OR_R64_TO_RM64 (0x09) 59 #define OPCODE_XOR_R64_TO_RM64 (0x31) 60 #define OPCODE_ADD_R64_TO_RM64 (0x01) 61 #define OPCODE_ADD_I32_TO_RM32 (0x81) 62 #define OPCODE_ADD_I8_TO_RM32 (0x83) 63 #define OPCODE_SUB_R64_FROM_RM64 (0x29) 64 #define OPCODE_SUB_I32_FROM_RM64 (0x81) 65 #define OPCODE_SUB_I8_FROM_RM64 (0x83) 69 #define OPCODE_SHL_RM64_CL (0xd3) 70 #define OPCODE_SAR_RM64_CL (0xd3) 73 #define OPCODE_CMP_R64_WITH_RM64 (0x39) 75 #define OPCODE_TEST_R8_WITH_RM8 (0x84) 76 #define OPCODE_JMP_REL8 (0xeb) 77 #define OPCODE_JMP_REL32 (0xe9) 78 #define OPCODE_JCC_REL8 (0x70) 79 #define OPCODE_JCC_REL32_A (0x0f) 80 #define OPCODE_JCC_REL32_B (0x80) 81 #define OPCODE_SETCC_RM8_A (0x0f) 82 #define OPCODE_SETCC_RM8_B (0x90) 83 #define OPCODE_CALL_REL32 (0xe8) 84 #define OPCODE_CALL_RM32 (0xff) 85 #define OPCODE_LEAVE (0xc9) 87 #define MODRM_R64(x) (((x) & 0x7) << 3) 88 #define MODRM_RM_DISP0 (0x00) 89 #define MODRM_RM_DISP8 (0x40) 90 #define MODRM_RM_DISP32 (0x80) 91 #define MODRM_RM_REG (0xc0) 92 #define MODRM_RM_R64(x) ((x) & 0x7) 94 #define OP_SIZE_PREFIX (0x66) 96 #define REX_PREFIX (0x40) 97 #define REX_W (0x08) // width 98 #define REX_R (0x04) // register 99 #define REX_X (0x02) // index 100 #define REX_B (0x01) // base 101 #define REX_W_FROM_R64(r64) ((r64) >> 0 & 0x08) 102 #define REX_R_FROM_R64(r64) ((r64) >> 1 & 0x04) 103 #define REX_X_FROM_R64(r64) ((r64) >> 2 & 0x02) 104 #define REX_B_FROM_R64(r64) ((r64) >> 3 & 0x01) 106 #define IMM32_L0(x) ((x) & 0xff) 107 #define IMM32_L1(x) (((x) >> 8) & 0xff) 108 #define IMM32_L2(x) (((x) >> 16) & 0xff) 109 #define IMM32_L3(x) (((x) >> 24) & 0xff) 110 #define IMM64_L4(x) (((x) >> 32) & 0xff) 111 #define IMM64_L5(x) (((x) >> 40) & 0xff) 112 #define IMM64_L6(x) (((x) >> 48) & 0xff) 113 #define IMM64_L7(x) (((x) >> 56) & 0xff) 115 #define UNSIGNED_FIT8(x) (((x) & 0xffffffffffffff00) == 0) 116 #define UNSIGNED_FIT32(x) (((x) & 0xffffffff00000000) == 0) 117 #define SIGNED_FIT8(x) (((x) & 0xffffff80) == 0) || (((x) & 0xffffff80) == 0xffffff80) 119 static inline byte *asm_x64_get_cur_to_write_bytes(
asm_x64_t *as,
int n) {
124 byte*
c = asm_x64_get_cur_to_write_bytes(as, 1);
131 byte*
c = asm_x64_get_cur_to_write_bytes(as, 2);
139 byte*
c = asm_x64_get_cur_to_write_bytes(as, 3);
148 byte*
c = asm_x64_get_cur_to_write_bytes(as, 4);
150 c[0] = IMM32_L0(w32);
151 c[1] = IMM32_L1(w32);
152 c[2] = IMM32_L2(w32);
153 c[3] = IMM32_L3(w32);
158 byte*
c = asm_x64_get_cur_to_write_bytes(as, 8);
160 c[0] = IMM32_L0(w64);
161 c[1] = IMM32_L1(w64);
162 c[2] = IMM32_L2(w64);
163 c[3] = IMM32_L3(w64);
164 c[4] = IMM64_L4(w64);
165 c[5] = IMM64_L5(w64);
166 c[6] = IMM64_L6(w64);
167 c[7] = IMM64_L7(w64);
183 STATIC void asm_x64_write_r64_disp(
asm_x64_t *as,
int r64,
int disp_r64,
int disp_offset) {
188 assert(SIGNED_FIT8(disp_offset));
189 asm_x64_write_byte_3(as, MODRM_R64(r64) | MODRM_RM_DISP8 | MODRM_RM_R64(disp_r64), 0x24, IMM32_L0(disp_offset));
194 asm_x64_write_byte_1(as, MODRM_R64(r64) | MODRM_RM_DISP0 | MODRM_RM_R64(disp_r64));
195 }
else if (SIGNED_FIT8(disp_offset)) {
196 asm_x64_write_byte_2(as, MODRM_R64(r64) | MODRM_RM_DISP8 | MODRM_RM_R64(disp_r64), IMM32_L0(disp_offset));
198 asm_x64_write_byte_1(as, MODRM_R64(r64) | MODRM_RM_DISP32 | MODRM_RM_R64(disp_r64));
199 asm_x64_write_word32(as, disp_offset);
203 STATIC void asm_x64_generic_r64_r64(
asm_x64_t *as,
int dest_r64,
int src_r64,
int op) {
204 asm_x64_write_byte_3(as, REX_PREFIX | REX_W | REX_R_FROM_R64(src_r64) | REX_B_FROM_R64(dest_r64), op, MODRM_R64(src_r64) | MODRM_RM_REG | MODRM_RM_R64(dest_r64));
208 asm_x64_write_byte_1(as, OPCODE_NOP);
213 asm_x64_write_byte_1(as, OPCODE_PUSH_R64 | src_r64);
215 asm_x64_write_byte_2(as, REX_PREFIX | REX_B, OPCODE_PUSH_R64 | (src_r64 & 7));
236 asm_x64_write_byte_1(as, OPCODE_POP_R64 | dest_r64);
238 asm_x64_write_byte_2(as, REX_PREFIX | REX_B, OPCODE_POP_R64 | (dest_r64 & 7));
243 asm_x64_write_byte_1(as, OPCODE_RET);
247 asm_x64_generic_r64_r64(as, dest_r64, src_r64, OPCODE_MOV_R64_TO_RM64);
251 if (src_r64 < 8 && dest_r64 < 8) {
252 asm_x64_write_byte_1(as, OPCODE_MOV_R8_TO_RM8);
254 asm_x64_write_byte_2(as, REX_PREFIX | REX_R_FROM_R64(src_r64) | REX_B_FROM_R64(dest_r64), OPCODE_MOV_R8_TO_RM8);
256 asm_x64_write_r64_disp(as, src_r64, dest_r64, dest_disp);
260 if (src_r64 < 8 && dest_r64 < 8) {
261 asm_x64_write_byte_2(as, OP_SIZE_PREFIX, OPCODE_MOV_R64_TO_RM64);
263 asm_x64_write_byte_3(as, OP_SIZE_PREFIX, REX_PREFIX | REX_R_FROM_R64(src_r64) | REX_B_FROM_R64(dest_r64), OPCODE_MOV_R64_TO_RM64);
265 asm_x64_write_r64_disp(as, src_r64, dest_r64, dest_disp);
269 if (src_r64 < 8 && dest_r64 < 8) {
270 asm_x64_write_byte_1(as, OPCODE_MOV_R64_TO_RM64);
272 asm_x64_write_byte_2(as, REX_PREFIX | REX_R_FROM_R64(src_r64) | REX_B_FROM_R64(dest_r64), OPCODE_MOV_R64_TO_RM64);
274 asm_x64_write_r64_disp(as, src_r64, dest_r64, dest_disp);
279 asm_x64_write_byte_2(as, REX_PREFIX | REX_W | REX_R_FROM_R64(src_r64) | REX_B_FROM_R64(dest_r64), OPCODE_MOV_R64_TO_RM64);
280 asm_x64_write_r64_disp(as, src_r64, dest_r64, dest_disp);
286 asm_x64_write_byte_2(as, 0x0f, OPCODE_MOVZX_RM8_TO_R64);
288 asm_x64_write_byte_3(as, REX_PREFIX | REX_R, 0x0f, OPCODE_MOVZX_RM8_TO_R64);
290 asm_x64_write_r64_disp(as, dest_r64, src_r64, src_disp);
296 asm_x64_write_byte_2(as, 0x0f, OPCODE_MOVZX_RM16_TO_R64);
298 asm_x64_write_byte_3(as, REX_PREFIX | REX_R, 0x0f, OPCODE_MOVZX_RM16_TO_R64);
300 asm_x64_write_r64_disp(as, dest_r64, src_r64, src_disp);
306 asm_x64_write_byte_1(as, OPCODE_MOV_RM64_TO_R64);
308 asm_x64_write_byte_2(as, REX_PREFIX | REX_R, OPCODE_MOV_RM64_TO_R64);
310 asm_x64_write_r64_disp(as, dest_r64, src_r64, src_disp);
315 asm_x64_write_byte_2(as, REX_PREFIX | REX_W | REX_R_FROM_R64(dest_r64) | REX_B_FROM_R64(src_r64), OPCODE_MOV_RM64_TO_R64);
316 asm_x64_write_r64_disp(as, dest_r64, src_r64, src_disp);
319 STATIC void asm_x64_lea_disp_to_r64(
asm_x64_t *as,
int src_r64,
int src_disp,
int dest_r64) {
323 asm_x64_write_byte_2(as, REX_PREFIX | REX_W, OPCODE_LEA_MEM_TO_R64);
324 asm_x64_write_r64_disp(as, dest_r64, src_r64, src_disp);
334 STATIC void asm_x64_mov_i32_to_r64(
asm_x64_t *as,
int src_i32,
int dest_r64) {
337 asm_x64_write_byte_1(as, OPCODE_MOV_I64_TO_R64 | dest_r64);
339 asm_x64_write_byte_2(as, REX_PREFIX | REX_B, OPCODE_MOV_I64_TO_R64 | (dest_r64 & 7));
341 asm_x64_write_word32(as, src_i32);
347 asm_x64_write_byte_2(as,
348 REX_PREFIX | REX_W | (dest_r64 < 8 ? 0 : REX_B),
349 OPCODE_MOV_I64_TO_R64 | (dest_r64 & 7));
350 asm_x64_write_word64(as, src_i64);
355 if (UNSIGNED_FIT32(src_i64)) {
357 asm_x64_mov_i32_to_r64(as, src_i64 & 0xffffffff, dest_r64);
374 asm_x64_generic_r64_r64(as, dest_r64, src_r64, OPCODE_AND_R64_TO_RM64);
378 asm_x64_generic_r64_r64(as, dest_r64, src_r64, OPCODE_OR_R64_TO_RM64);
382 asm_x64_generic_r64_r64(as, dest_r64, src_r64, OPCODE_XOR_R64_TO_RM64);
386 asm_x64_generic_r64_r64(as, dest_r64, 4, OPCODE_SHL_RM64_CL);
390 asm_x64_generic_r64_r64(as, dest_r64, 7, OPCODE_SAR_RM64_CL);
394 asm_x64_generic_r64_r64(as, dest_r64, src_r64, OPCODE_ADD_R64_TO_RM64);
398 asm_x64_generic_r64_r64(as, dest_r64, src_r64, OPCODE_SUB_R64_FROM_RM64);
403 asm_x64_write_byte_1(as, REX_PREFIX | REX_W | REX_R_FROM_R64(dest_r64) | REX_B_FROM_R64(src_r64));
404 asm_x64_write_byte_3(as, 0x0f, 0xaf, MODRM_R64(dest_r64) | MODRM_RM_REG | MODRM_RM_R64(src_r64));
421 STATIC void asm_x64_sub_r64_i32(
asm_x64_t *as,
int dest_r64,
int src_i32) {
423 if (SIGNED_FIT8(src_i32)) {
425 asm_x64_write_byte_3(as, REX_PREFIX | REX_W, OPCODE_SUB_I8_FROM_RM64, MODRM_R64(5) | MODRM_RM_REG | MODRM_RM_R64(dest_r64));
426 asm_x64_write_byte_1(as, src_i32 & 0xff);
429 asm_x64_write_byte_3(as, REX_PREFIX | REX_W, OPCODE_SUB_I32_FROM_RM64, MODRM_R64(5) | MODRM_RM_REG | MODRM_RM_R64(dest_r64));
430 asm_x64_write_word32(as, src_i32);
452 asm_x64_generic_r64_r64(as, src_r64_b, src_r64_a, OPCODE_CMP_R64_WITH_RM64);
471 asm_x64_write_byte_2(as, OPCODE_TEST_R8_WITH_RM8, MODRM_R64(src_r64_a) | MODRM_RM_REG | MODRM_RM_R64(src_r64_b));
476 asm_x64_write_byte_3(as, OPCODE_SETCC_RM8_A, OPCODE_SETCC_RM8_B | jcc_type, MODRM_R64(0) | MODRM_RM_REG | MODRM_RM_R64(dest_r8));
480 assert(label < as->base.max_num_labels);
485 mp_uint_t dest = get_label_dest(as, label);
491 if (SIGNED_FIT8(rel)) {
492 asm_x64_write_byte_2(as, OPCODE_JMP_REL8, rel & 0xff);
501 asm_x64_write_byte_1(as, OPCODE_JMP_REL32);
502 asm_x64_write_word32(as, rel);
507 mp_uint_t dest = get_label_dest(as, label);
513 if (SIGNED_FIT8(rel)) {
514 asm_x64_write_byte_2(as, OPCODE_JCC_REL8 | jcc_type, rel & 0xff);
523 asm_x64_write_byte_2(as, OPCODE_JCC_REL32_A, OPCODE_JCC_REL32_B | jcc_type);
524 asm_x64_write_word32(as, rel);
531 if (num_locals < 0) {
546 asm_x64_write_byte_1(as, OPCODE_LEAVE);
561 STATIC int asm_x64_local_offset_from_ebp(
asm_x64_t *as,
int local_num) {
562 return (-as->
num_locals + local_num) * WORD_SIZE;
574 int offset = asm_x64_local_offset_from_ebp(as, local_num);
623 asm_x64_write_byte_2(as, OPCODE_CALL_RM32, MODRM_R64(2) | MODRM_RM_REG | MODRM_RM_R64(temp_r64));
632 #endif // MICROPY_EMIT_X64 void asm_x64_shl_r64_cl(asm_x64_t *as, int dest_r64)
void asm_x64_cmp_r64_with_r64(asm_x64_t *as, int src_r64_a, int src_r64_b)
uint8_t * mp_asm_base_get_cur_to_write_bytes(mp_asm_base_t *as, size_t num_bytes_to_write)
void asm_x64_mov_mem8_to_r64zx(asm_x64_t *as, int src_r64, int src_disp, int dest_r64)
void asm_x64_mov_r64_to_local(asm_x64_t *as, int src_r64, int dest_local_num)
void asm_x64_sub_r64_r64(asm_x64_t *as, int dest_r64, int src_r64)
void asm_x64_call_ind(asm_x64_t *as, void *ptr, int temp_r32)
void asm_x64_mov_r32_to_mem32(asm_x64_t *as, int src_r64, int dest_r64, int dest_disp)
void asm_x64_jcc_label(asm_x64_t *as, int jcc_type, mp_uint_t label)
void asm_x64_mul_r64_r64(asm_x64_t *as, int dest_r64, int src_r64)
void asm_x64_pop_r64(asm_x64_t *as, int dest_r64)
void asm_x64_setcc_r8(asm_x64_t *as, int jcc_type, int dest_r8)
void asm_x64_nop(asm_x64_t *as)
void asm_x64_mov_i64_to_r64(asm_x64_t *as, int64_t src_i64, int dest_r64)
void asm_x64_mov_r64_r64(asm_x64_t *as, int dest_r64, int src_r64)
void asm_x64_add_r64_r64(asm_x64_t *as, int dest_r64, int src_r64)
void asm_x64_entry(asm_x64_t *as, int num_locals)
void asm_x64_mov_i64_to_r64_optimised(asm_x64_t *as, int64_t src_i64, int dest_r64)
void asm_x64_xor_r64_r64(asm_x64_t *as, int dest_r64, int src_r64)
void asm_x64_push_r64(asm_x64_t *as, int src_r64)
void asm_x64_test_r8_with_r8(asm_x64_t *as, int src_r64_a, int src_r64_b)
void asm_x64_exit(asm_x64_t *as)
void asm_x64_jmp_label(asm_x64_t *as, mp_uint_t label)
void asm_x64_mov_r16_to_mem16(asm_x64_t *as, int src_r64, int dest_r64, int dest_disp)
void asm_x64_mov_local_addr_to_r64(asm_x64_t *as, int local_num, int dest_r64)
void asm_x64_mov_mem64_to_r64(asm_x64_t *as, int src_r64, int src_disp, int dest_r64)
void asm_x64_sar_r64_cl(asm_x64_t *as, int dest_r64)
void asm_x64_mov_mem32_to_r64zx(asm_x64_t *as, int src_r64, int src_disp, int dest_r64)
void asm_x64_mov_r8_to_mem8(asm_x64_t *as, int src_r64, int dest_r64, int dest_disp)
void asm_x64_mov_i64_to_r64_aligned(asm_x64_t *as, int64_t src_i64, int dest_r64)
void asm_x64_mov_mem16_to_r64zx(asm_x64_t *as, int src_r64, int src_disp, int dest_r64)
void asm_x64_or_r64_r64(asm_x64_t *as, int dest_r64, int src_r64)
void asm_x64_and_r64_r64(asm_x64_t *as, int dest_r64, int src_r64)
void asm_x64_mov_r64_to_mem64(asm_x64_t *as, int src_r64, int dest_r64, int dest_disp)
void asm_x64_mov_local_to_r64(asm_x64_t *as, int src_local_num, int dest_r64)