clipper.c
来自「gcc3.2.1源代码」· C语言 代码 · 共 693 行 · 第 1/2 页
C
693 行
output_addr_const (file, addr); }}const char *rev_cond_name (op) rtx op;{ switch (GET_CODE (op)) { case EQ: return "ne"; case NE: return "eq"; case LT: return "ge"; case LE: return "gt"; case GT: return "le"; case GE: return "lt"; case LTU: return "geu"; case LEU: return "gtu"; case GTU: return "leu"; case GEU: return "ltu"; default: abort (); }}/* Dump the argument register to the stack; return the location of the block. */struct rtx_def *clipper_builtin_saveregs (){ rtx block, addr, r0_addr, r1_addr, f0_addr, f1_addr, mem; int set = get_varargs_alias_set (); /* Allocate the save area for r0,r1,f0,f1 */ block = assign_stack_local (BLKmode, 6 * UNITS_PER_WORD, 2 * BITS_PER_WORD); RTX_UNCHANGING_P (block) = 1; RTX_UNCHANGING_P (XEXP (block, 0)) = 1; addr = XEXP (block, 0); r0_addr = addr; r1_addr = plus_constant (addr, 4); f0_addr = plus_constant (addr, 8); f1_addr = plus_constant (addr, 16); /* Store int regs */ mem = gen_rtx_MEM (SImode, r0_addr); set_mem_alias_set (mem, set); emit_move_insn (mem, gen_rtx_REG (SImode, 0)); mem = gen_rtx_MEM (SImode, r1_addr); set_mem_alias_set (mem, set); emit_move_insn (mem, gen_rtx_REG (SImode, 1)); /* Store float regs */ mem = gen_rtx_MEM (DFmode, f0_addr); set_mem_alias_set (mem, set); emit_move_insn (mem, gen_rtx_REG (DFmode, 16)); mem = gen_rtx_MEM (DFmode, f1_addr); set_mem_alias_set (mem, set); emit_move_insn (mem, gen_rtx_REG (DFmode, 17)); return addr;}treeclipper_build_va_list (){ tree record, ap, reg, num; /* struct { int __va_ap; // pointer to stack args void *__va_reg[4]; // pointer to r0,f0,r1,f1 int __va_num; // number of args processed }; */ record = make_node (RECORD_TYPE); num = build_decl (FIELD_DECL, get_identifier ("__va_num"), integer_type_node); DECL_FIELD_CONTEXT (num) = record; reg = build_decl (FIELD_DECL, get_identifier ("__va_reg"), build_array_type (ptr_type_node, build_index_type (build_int_2 (3, 0)))); DECL_FIELD_CONTEXT (reg) = record; TREE_CHAIN (reg) = num; ap = build_decl (FIELD_DECL, get_identifier ("__va_ap"), integer_type_node); DECL_FIELD_CONTEXT (ap) = record; TREE_CHAIN (ap) = reg; TYPE_FIELDS (record) = ap; layout_type (record); return record;}voidclipper_va_start (stdarg_p, valist, nextarg) int stdarg_p; tree valist; rtx nextarg ATTRIBUTE_UNUSED;{ tree ap_field, reg_field, num_field; tree t, u, save_area; ap_field = TYPE_FIELDS (TREE_TYPE (valist)); reg_field = TREE_CHAIN (ap_field); num_field = TREE_CHAIN (reg_field); ap_field = build (COMPONENT_REF, TREE_TYPE (ap_field), valist, ap_field); reg_field = build (COMPONENT_REF, TREE_TYPE (reg_field), valist, reg_field); num_field = build (COMPONENT_REF, TREE_TYPE (num_field), valist, num_field); /* Call __builtin_saveregs to save r0, r1, f0, and f1 in a block. */ save_area = make_tree (integer_type_node, expand_builtin_saveregs ()); /* Set __va_ap. */ t = make_tree (ptr_type_node, virtual_incoming_args_rtx); if (stdarg_p && current_function_args_info.size != 0) t = build (PLUS_EXPR, ptr_type_node, t, build_int_2 (current_function_args_info.size, 0)); t = build (MODIFY_EXPR, TREE_TYPE (ap_field), ap_field, t); TREE_SIDE_EFFECTS (t) = 1; expand_expr (t, const0_rtx, VOIDmode, EXPAND_NORMAL); /* Set the four entries of __va_reg. */ t = build1 (NOP_EXPR, ptr_type_node, save_area); u = build (ARRAY_REF, ptr_type_node, reg_field, build_int_2 (0, 0)); t = build (MODIFY_EXPR, ptr_type_node, u, t); TREE_SIDE_EFFECTS (t) = 1; expand_expr (t, const0_rtx, VOIDmode, EXPAND_NORMAL); t = fold (build (PLUS_EXPR, integer_type_node, save_area, build_int_2 (8, 0))); t = build1 (NOP_EXPR, ptr_type_node, save_area); u = build (ARRAY_REF, ptr_type_node, reg_field, build_int_2 (1, 0)); t = build (MODIFY_EXPR, ptr_type_node, u, t); TREE_SIDE_EFFECTS (t) = 1; expand_expr (t, const0_rtx, VOIDmode, EXPAND_NORMAL); t = fold (build (PLUS_EXPR, integer_type_node, save_area, build_int_2 (4, 0))); t = build1 (NOP_EXPR, ptr_type_node, save_area); u = build (ARRAY_REF, ptr_type_node, reg_field, build_int_2 (2, 0)); t = build (MODIFY_EXPR, ptr_type_node, u, t); TREE_SIDE_EFFECTS (t) = 1; expand_expr (t, const0_rtx, VOIDmode, EXPAND_NORMAL); t = fold (build (PLUS_EXPR, integer_type_node, save_area, build_int_2 (16, 0))); t = build1 (NOP_EXPR, ptr_type_node, save_area); u = build (ARRAY_REF, ptr_type_node, reg_field, build_int_2 (3, 0)); t = build (MODIFY_EXPR, ptr_type_node, u, t); TREE_SIDE_EFFECTS (t) = 1; expand_expr (t, const0_rtx, VOIDmode, EXPAND_NORMAL); /* Set __va_num. */ t = build_int_2 (current_function_args_info.num, 0); t = build (MODIFY_EXPR, TREE_TYPE (num_field), num_field, t); TREE_SIDE_EFFECTS (t) = 1; expand_expr (t, const0_rtx, VOIDmode, EXPAND_NORMAL);}rtxclipper_va_arg (valist, type) tree valist, type;{ tree ap_field, reg_field, num_field; tree addr, t; HOST_WIDE_INT align; rtx addr_rtx, over_label = NULL_RTX, tr; /* Integers: if (VA.__va_num < 2) addr = VA.__va_reg[2 * VA.__va_num]; else addr = round(VA.__va_ap), VA.__va_ap = round(VA.__va_ap) + sizeof(TYPE); VA.__va_num++; Floats: if (VA.__va_num < 2) addr = VA.__va_reg[2 * VA.__va_num + 1]; else addr = round(VA.__va_ap), VA.__va_ap = round(VA.__va_ap) + sizeof(TYPE); VA.__va_num++; Aggregates: addr = round(VA.__va_ap), VA.__va_ap = round(VA.__va_ap) + sizeof(TYPE); VA.__va_num++; */ ap_field = TYPE_FIELDS (TREE_TYPE (valist)); reg_field = TREE_CHAIN (ap_field); num_field = TREE_CHAIN (reg_field); ap_field = build (COMPONENT_REF, TREE_TYPE (ap_field), valist, ap_field); reg_field = build (COMPONENT_REF, TREE_TYPE (reg_field), valist, reg_field); num_field = build (COMPONENT_REF, TREE_TYPE (num_field), valist, num_field); addr_rtx = gen_reg_rtx (Pmode); if (! AGGREGATE_TYPE_P (type)) { tree inreg; rtx false_label; over_label = gen_label_rtx (); false_label = gen_label_rtx (); emit_cmp_and_jump_insns (expand_expr (num_field, NULL_RTX, 0, OPTAB_LIB_WIDEN), GEN_INT (2), GE, const0_rtx, TYPE_MODE (TREE_TYPE (num_field)), TREE_UNSIGNED (num_field), false_label); inreg = fold (build (MULT_EXPR, integer_type_node, num_field, build_int_2 (2, 0))); if (FLOAT_TYPE_P (type)) inreg = fold (build (PLUS_EXPR, integer_type_node, inreg, build_int_2 (1, 0))); inreg = fold (build (ARRAY_REF, ptr_type_node, reg_field, inreg)); tr = expand_expr (inreg, addr_rtx, VOIDmode, EXPAND_NORMAL); if (tr != addr_rtx) emit_move_insn (addr_rtx, tr); emit_jump_insn (gen_jump (over_label)); emit_barrier (); emit_label (false_label); } /* Round to alignment of `type', or at least integer alignment. */ align = TYPE_ALIGN (type); if (align < TYPE_ALIGN (integer_type_node)) align = TYPE_ALIGN (integer_type_node); align /= BITS_PER_UNIT; addr = fold (build (PLUS_EXPR, ptr_type_node, ap_field, build_int_2 (align-1, 0))); addr = fold (build (BIT_AND_EXPR, ptr_type_node, addr, build_int_2 (-align, -1))); addr = save_expr (addr); tr = expand_expr (addr, addr_rtx, Pmode, EXPAND_NORMAL); if (tr != addr_rtx) emit_move_insn (addr_rtx, tr); t = build (MODIFY_EXPR, TREE_TYPE (ap_field), ap_field, build (PLUS_EXPR, TREE_TYPE (ap_field), addr, build_int_2 (int_size_in_bytes (type), 0))); TREE_SIDE_EFFECTS (t) = 1; expand_expr (t, const0_rtx, VOIDmode, EXPAND_NORMAL); if (over_label) emit_label (over_label); t = build (MODIFY_EXPR, TREE_TYPE (num_field), num_field, build (PLUS_EXPR, TREE_TYPE (num_field), num_field, build_int_2 (1, 0))); TREE_SIDE_EFFECTS (t) = 1; expand_expr (t, const0_rtx, VOIDmode, EXPAND_NORMAL); return addr_rtx;}/* Return truth value of whether OP can be used as an word register operand. Reject (SUBREG:SI (REG:SF )) */intint_reg_operand (op, mode) rtx op; enum machine_mode mode;{ return (register_operand (op, mode) && (GET_CODE (op) != SUBREG || GET_MODE_CLASS (GET_MODE (SUBREG_REG (op))) == MODE_INT));}/* Return truth value of whether OP can be used as a float register operand. Reject (SUBREG:SF (REG:SI )) )) */intfp_reg_operand (op, mode) rtx op; enum machine_mode mode;{ return (register_operand (op, mode) && (GET_CODE (op) != SUBREG || GET_MODE_CLASS (GET_MODE (SUBREG_REG (op))) == MODE_FLOAT));}static voidclix_asm_out_constructor (symbol, priority) rtx symbol; int priority ATTRIBUTE_UNUSED;{ init_section (); fputs ("\tloada ", asm_out_file); assemble_name (asm_out_file, XSTR (symbol, 0)); fputs (",r0\n\tsubq $8,sp\n\tstorw r0,(sp)\n", asm_out_file);}static voidclix_asm_out_destructor (symbol, priority) rtx symbol; int priority ATTRIBUTE_UNUSED;{ fini_section (); assemble_integer (symbol, POINTER_SIZE / BITS_PER_UNIT, POINTER_SIZE, 1); assemble_integer (const0_rtx, POINTER_SIZE / BITS_PER_UNIT, POINTER_SIZE, 1);}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?