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.