Hi,

I'm writing a backend for gcc3.4.6 and I'm a bit confused about reload passes.



-> my constraints :

'b' = arithmetic/logical unit register ($R, class R_REGS)
'c' = counter register ($C, class C_REGS)
'v' = memory with several special constraint (also described in predicate 
my_mem_or_reg_operand)



-> movsi expand/insn :

 (define_insn "movsi_internal"
  [(set    
      (match_operand:SI 0 "my_mem_or_reg_operand" "=v,v,c,b,b ")
      (match_operand: SI 1 "general_operand" "c,b,v,v,i "))]
  ""
  "@
  mov %1,%0
  [.]
  loadi %1,%0"
)

 (define_expand "movsi"
  [(set    
      (match_operand:SI 0 "general_operand" "")
      (match_operand:SI 1 "general_operand" ""))]
  ""
  {
     /* Handle Memory to Memory and immediate to memory movs */
     if (((GET_CODE (operands[1]) == MEM) || immediate_operand (operands[1], 
SImode))
          && (GET_CODE(operands[0])== MEM) )
     { 
        operands[1] = force_reg(SImode,operands[1]);
     }
  }
)

Note there is no instruction to move immediate to $R regs ('b' constraint) but 
there is one to move immediate to $C regs ('c' constraint).

I have no mean to express this limitation inside predicates (because pseudo 
register don't known in which hard register they will be allocated), reload 
pass create some moves from immediate to $R.  :


                arithmetic.c:197: error: insn does not satisfy its constraints:
                (insn 1505 903 1506 0 (set (reg:SI 25 $R9)
                        (const_int 8 [0x8])) 0 {movsi_internal} (nil)
                    (nil))
                arithmetic.c:197: internal compiler error: in 
reload_cse_simplify_operands, at postreload.c:391





So, I read the doc :) and figured out how to control the reload pass. I added 2 
MACROs and the reload_insi expand :

#define PREFERRED_RELOAD_CLASS(X,CLASS) my_preferred_reload_class(X,CLASS)
#define SECONDARY_INPUT_RELOAD_CLASS(CLASS, MODE, X) 
my_secondary_input_reload_class (CLASS, MODE, X)

enum reg_class my_preferred_reload_class(rtx x, enum reg_class rclass)
{
   if (GET_CODE(x) == CONST_INT || GET_CODE(x) == CONST_DOUBLE) {
       if (rclass == GENERAL_REGS) {
          return C_REGS;
       }
       else if (rclass == R_REGS) {
          return NO_REGS; /* put the constant in memory */
       }
   }
   return rclass;
}

enum reg_class my_secondary_input_reload_class(enum reg_class rclass, enum 
machine_mode mode, rtx x)
{
  if ((GET_CODE(x) == CONST_INT || GET_CODE(x) == CONST_DOUBLE) && rclass == 
R_REGS) {
     return C_REGS; /* use C_REGS for clobber in reload_insi */
  }
  return NO_REGS;
}

(define_expand "reload_insi"
  [(set 
      (match_operand:SI 0 "register_operand" "=b")
      (match_operand:SI 1 "immediate_operand" "i"))
   (clobber (match_operand:SI 2 "register_operand" "=c"))]
    "
    {            
       emit_insn(gen_rtx_SET(SImode, operands[2], operands[1]));
       emit_insn(gen_rtx_SET(SImode, operands[0], operands[2]));
       DONE;
    }
)

Theses MACROs and insn do not solve my problem and the forbidden move keeps on 
being emitted.
Did I misunderstood something ? A my doing something illegal ? Should I do 
things differently?
(A printf in my reload_insi shows that this expand is never called.)

                Thank you for your help.

Reply via email to