- 添加 scc_ast2ir_mir_module 内联函数统一访问模块 - 替换所有直接访问 ctx->builder.cprog->module 的地方 - 移除重复的 scc_hir_type_size 函数实现 - 添加 scc_hir_module_type_size 函数到模块接口 - 更新所有类型大小计算调用使用新函数 feat(hir): 增强构建器安全性和全局变量处理 - 为 scc_hir_builder_integer 添加空指针检查断言 - 修复 scc_hir_builder_global_alloca 中全局变量类型设置 - 改进 scc_hir_builder_get_elem_ptr 处理空指针索引情况 - 重构字符串常量生成使用 get_elem_ptr 构建器函数 refactor(lir): 简化地址表达式表示并增强内置函数支持 - 移除复杂地址结构体 scc_lir_addr_t - 简化 scc_lir_instr 结构体中的地址表示 - 移除 STORE_ADDR 操作码 - 添加 memcpy 和 memset 内置函数操作码 - 在符号元数据中使用联合体替代嵌套结构体 feat(hir2lir): 完善 HIR 到 LIR 转换中的内置函数处理 - 添加 ensure_vreg 辅助函数确保虚拟寄存器操作数 - 正确处理全局变量地址符号引用 - 优化 GET_ELEM_PTR 转换使用类型大小计算 - 完整实现所有内置函数(BUILTIN)的 LIR 转换 - 包括 memcpy、memset、va_start、va_arg、va_end、va_copy 等
109 lines
3.9 KiB
C
109 lines
3.9 KiB
C
#ifndef __SCC_X86_MIR_H__
|
||
#define __SCC_X86_MIR_H__
|
||
|
||
#include "../scc_mir.h"
|
||
#include <scc_cfg.h>
|
||
#include <scc_pos.h>
|
||
#include <x86/scc_x86_encode.h>
|
||
#include <x86/scc_x86_iform.h>
|
||
#include <x86/scc_x86_reg.h>
|
||
|
||
typedef struct {
|
||
union {
|
||
int opcode;
|
||
scc_x86_iform_t iform;
|
||
};
|
||
u8 num_operands;
|
||
scc_x86_operand_value_t operands[6];
|
||
scc_pos_t src_loc;
|
||
} scc_x86_instr_t;
|
||
|
||
// x86 后端指令:首字段 int opcode(正 = scc_x86_iform_t,负 = 伪指令)
|
||
typedef union scc_mir_x86_instr {
|
||
scc_mir_instr_t instr;
|
||
scc_x86_instr_t x86_instr;
|
||
} scc_mir_x86_instr_t;
|
||
|
||
typedef SCC_VEC(scc_mir_x86_instr_t) scc_mir_x86_instr_vec_t;
|
||
|
||
// ── 基本块 values 强制转换 ──────────────────────────────────────────────
|
||
|
||
#define SCC_MIR_X86_BBLOCK_INSTRS(bb) ((scc_mir_x86_instr_vec_t *)&bb->values)
|
||
#define SCC_MIR_X86_BBLOCK_INSTRS_C(bb) \
|
||
((const scc_mir_x86_instr_vec_t *)&bb->values)
|
||
|
||
static inline cbool scc_x86_reg_is_vreg(scc_x86_reg_t reg) {
|
||
return reg >= SCC_X86_REG_COUNT;
|
||
}
|
||
static inline cbool scc_x86_op_is_vreg(const scc_x86_operand_value_t *op) {
|
||
return op->kind == SCC_X86_OPR_REG && scc_x86_reg_is_vreg(op->reg);
|
||
}
|
||
static inline int scc_x86_reg_get_vreg(scc_x86_reg_t reg) {
|
||
return (int)reg - (int)SCC_X86_REG_COUNT;
|
||
}
|
||
static inline int scc_x86_op_get_vreg(const scc_x86_operand_value_t *op) {
|
||
return scc_x86_reg_get_vreg(op->reg);
|
||
}
|
||
static inline void scc_x86_op_set_preg(scc_x86_operand_value_t *op,
|
||
scc_x86_reg_t preg) {
|
||
op->kind = SCC_X86_OPR_REG;
|
||
op->reg = preg;
|
||
}
|
||
|
||
// slot_id 编码为 base=INVALID, disp=slot_id
|
||
static inline scc_x86_operand_value_t scc_x86_op_slot(int slot_id) {
|
||
scc_x86_operand_value_t o = {.kind = SCC_X86_OPR_MEM};
|
||
o.mem.base = SCC_X86_REG_INVALID;
|
||
o.mem.index = SCC_X86_REG_INVALID;
|
||
o.mem.scale = 1;
|
||
o.mem.disp.displacement = slot_id;
|
||
o.mem.disp.displacement_bits = 0;
|
||
return o;
|
||
}
|
||
|
||
static inline bool scc_x86_op_is_slot(const scc_x86_operand_value_t *op) {
|
||
return op->kind == SCC_X86_OPR_MEM && op->mem.base == SCC_X86_REG_INVALID;
|
||
}
|
||
static inline int scc_x86_op_slot_id(const scc_x86_operand_value_t *op) {
|
||
return op->mem.disp.displacement;
|
||
}
|
||
|
||
static inline void scc_mir_x86_instr_0(scc_mir_x86_instr_t *out, int opcode,
|
||
scc_pos_t pos) {
|
||
out->x86_instr.opcode = opcode;
|
||
out->x86_instr.num_operands = 0;
|
||
out->x86_instr.src_loc = pos;
|
||
}
|
||
static inline void scc_mir_x86_instr_1(scc_mir_x86_instr_t *out, int opcode,
|
||
scc_x86_operand_value_t op0,
|
||
scc_pos_t pos) {
|
||
out->x86_instr.opcode = opcode;
|
||
out->x86_instr.num_operands = 1;
|
||
out->x86_instr.operands[0] = op0;
|
||
out->x86_instr.src_loc = pos;
|
||
}
|
||
static inline void scc_mir_x86_instr_2(scc_mir_x86_instr_t *out, int opcode,
|
||
scc_x86_operand_value_t op0,
|
||
scc_x86_operand_value_t op1,
|
||
scc_pos_t pos) {
|
||
out->x86_instr.opcode = opcode;
|
||
out->x86_instr.num_operands = 2;
|
||
out->x86_instr.operands[0] = op0;
|
||
out->x86_instr.operands[1] = op1;
|
||
out->x86_instr.src_loc = pos;
|
||
}
|
||
static inline void scc_mir_x86_instr_3(scc_mir_x86_instr_t *out, int opcode,
|
||
scc_x86_operand_value_t op0,
|
||
scc_x86_operand_value_t op1,
|
||
scc_x86_operand_value_t op2,
|
||
scc_pos_t pos) {
|
||
out->x86_instr.opcode = opcode;
|
||
out->x86_instr.num_operands = 3;
|
||
out->x86_instr.operands[0] = op0;
|
||
out->x86_instr.operands[1] = op1;
|
||
out->x86_instr.operands[2] = op2;
|
||
out->x86_instr.src_loc = pos;
|
||
}
|
||
|
||
#endif /* __SCC_X86_MIR_H__ */
|