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 + -
显示快捷键?