mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-10-24 17:41:41 +02:00

* support conditional branches with 2 parameters * factored out common parts * adjusted range limits, they were copy-pasted from PowerPC target which uses 14-bit offsets. MIPS however uses 16-bit offsets, i.e. can branch 4 times farther. git-svn-id: trunk@33088 -
676 lines
19 KiB
ObjectPascal
676 lines
19 KiB
ObjectPascal
{
|
|
Copyright (c) 1999-2009 by Mazen Neifer and David Zhang
|
|
|
|
Contains the assembler object for the MIPSEL
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
****************************************************************************
|
|
}
|
|
unit aasmcpu;
|
|
|
|
{$i fpcdefs.inc}
|
|
|
|
interface
|
|
|
|
uses
|
|
cclasses,
|
|
globtype, globals, verbose,
|
|
aasmbase, aasmdata, aasmsym, aasmtai,
|
|
cgbase, cgutils, cpubase, cpuinfo;
|
|
|
|
const
|
|
{ "mov reg,reg" source operand number }
|
|
O_MOV_SOURCE = 1;
|
|
{ "mov reg,reg" source operand number }
|
|
O_MOV_DEST = 0;
|
|
|
|
type
|
|
{ taicpu }
|
|
taicpu = class(tai_cpu_abstract_sym)
|
|
constructor op_none(op: tasmop);
|
|
|
|
constructor op_reg(op: tasmop; _op1: tregister);
|
|
constructor op_const(op: tasmop; _op1: longint);
|
|
constructor op_ref(op: tasmop; const _op1: treference);
|
|
|
|
constructor op_reg_reg(op: tasmop; _op1, _op2: tregister);
|
|
constructor op_reg_ref(op: tasmop; _op1: tregister; const _op2: treference);
|
|
constructor op_reg_const(op: tasmop; _op1: tregister; _op2: longint);
|
|
constructor op_const_const(op: tasmop; _op1: aint; _op2: aint);
|
|
|
|
constructor op_reg_reg_reg(op: tasmop; _op1, _op2, _op3: tregister);
|
|
|
|
constructor op_reg_reg_ref(op: tasmop; _op1, _op2: tregister; const _op3: treference);
|
|
constructor op_reg_reg_const(op: tasmop; _op1, _op2: tregister; _op3: aint);
|
|
{ INS and EXT }
|
|
constructor op_reg_reg_const_const(op: tasmop; _op1,_op2: tregister; _op3,_op4: aint);
|
|
constructor op_reg_const_reg(op: tasmop; _op1: tregister; _op2: aint; _op3: tregister);
|
|
|
|
{ this is for Jmp instructions }
|
|
constructor op_sym(op: tasmop; _op1: tasmsymbol);
|
|
constructor op_reg_reg_sym(op: tasmop; _op1, _op2: tregister; _op3: tasmsymbol);
|
|
constructor op_reg_sym(op: tasmop; _op1: tregister; _op2: tasmsymbol);
|
|
constructor op_sym_ofs(op: tasmop; _op1: tasmsymbol; _op1ofs: longint);
|
|
|
|
{ register allocation }
|
|
function is_same_reg_move(regtype: Tregistertype): boolean; override;
|
|
|
|
{ register spilling code }
|
|
function spilling_get_operation_type(opnr: longint): topertype; override;
|
|
|
|
function is_macro: boolean;
|
|
end;
|
|
|
|
tai_align = class(tai_align_abstract)
|
|
{ nothing to add }
|
|
end;
|
|
|
|
procedure InitAsm;
|
|
procedure DoneAsm;
|
|
|
|
procedure fixup_jmps(list: TAsmList);
|
|
|
|
function spilling_create_load(const ref: treference; r: tregister): taicpu;
|
|
function spilling_create_store(r: tregister; const ref: treference): taicpu;
|
|
|
|
implementation
|
|
|
|
uses
|
|
cutils;
|
|
|
|
{*****************************************************************************
|
|
taicpu Constructors
|
|
*****************************************************************************}
|
|
|
|
constructor taicpu.op_none(op: tasmop);
|
|
begin
|
|
inherited Create(op);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_reg(op: tasmop; _op1: tregister);
|
|
begin
|
|
inherited Create(op);
|
|
ops := 1;
|
|
loadreg(0, _op1);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_ref(op: tasmop; const _op1: treference);
|
|
begin
|
|
inherited Create(op);
|
|
ops := 1;
|
|
loadref(0, _op1);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_const(op: tasmop; _op1: longint);
|
|
begin
|
|
inherited Create(op);
|
|
ops := 1;
|
|
loadconst(0, _op1);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_reg_reg(op: tasmop; _op1, _op2: tregister);
|
|
begin
|
|
inherited Create(op);
|
|
ops := 2;
|
|
loadreg(0, _op1);
|
|
loadreg(1, _op2);
|
|
end;
|
|
|
|
constructor taicpu.op_reg_const(op: tasmop; _op1: tregister; _op2: longint);
|
|
begin
|
|
inherited Create(op);
|
|
ops := 2;
|
|
loadreg(0, _op1);
|
|
loadconst(1, _op2);
|
|
end;
|
|
|
|
constructor taicpu.op_const_const(op: tasmop; _op1: aint; _op2: aint);
|
|
begin
|
|
inherited Create(op);
|
|
ops := 2;
|
|
loadconst(0, _op1);
|
|
loadconst(1, _op2);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_reg_ref(op: tasmop; _op1: tregister; const _op2: treference);
|
|
begin
|
|
inherited Create(op);
|
|
ops := 2;
|
|
loadreg(0, _op1);
|
|
loadref(1, _op2);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_reg_reg_reg(op: tasmop; _op1, _op2, _op3: tregister);
|
|
begin
|
|
inherited Create(op);
|
|
ops := 3;
|
|
loadreg(0, _op1);
|
|
loadreg(1, _op2);
|
|
loadreg(2, _op3);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_reg_reg_ref(op: tasmop; _op1, _op2: tregister; const _op3: treference);
|
|
begin
|
|
inherited create(op);
|
|
ops := 3;
|
|
loadreg(0, _op1);
|
|
loadreg(1, _op2);
|
|
loadref(2, _op3);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_reg_reg_const(op: tasmop; _op1, _op2: tregister; _op3: aint);
|
|
begin
|
|
inherited create(op);
|
|
ops := 3;
|
|
loadreg(0, _op1);
|
|
loadreg(1, _op2);
|
|
loadconst(2, _op3);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_reg_reg_const_const(op: tasmop; _op1, _op2: tregister; _op3, _op4: aint);
|
|
begin
|
|
inherited create(op);
|
|
ops := 4;
|
|
loadreg(0, _op1);
|
|
loadreg(1, _op2);
|
|
loadconst(2, _op3);
|
|
loadconst(3, _op4);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_reg_const_reg(op: tasmop; _op1: tregister; _op2: aint;
|
|
_op3: tregister);
|
|
begin
|
|
inherited create(op);
|
|
ops := 3;
|
|
loadreg(0, _op1);
|
|
loadconst(1, _op2);
|
|
loadreg(2, _op3);
|
|
end;
|
|
|
|
|
|
constructor taicpu.op_sym(op: tasmop; _op1: tasmsymbol);
|
|
begin
|
|
inherited Create(op);
|
|
is_jmp := op in [A_BC, A_BA];
|
|
ops := 1;
|
|
loadsymbol(0, _op1, 0);
|
|
end;
|
|
|
|
constructor taicpu.op_reg_reg_sym(op: tasmop; _op1, _op2: tregister; _op3: tasmsymbol);
|
|
begin
|
|
inherited create(op);
|
|
is_jmp := op in [A_BC, A_BA];
|
|
ops := 3;
|
|
loadreg(0, _op1);
|
|
loadreg(1, _op2);
|
|
loadsymbol(2, _op3, 0);
|
|
end;
|
|
|
|
constructor taicpu.op_reg_sym(op: tasmop; _op1: tregister; _op2: tasmsymbol);
|
|
begin
|
|
inherited create(op);
|
|
is_jmp := op in [A_BC, A_BA];
|
|
ops := 2;
|
|
loadreg(0, _op1);
|
|
loadsymbol(1, _op2, 0);
|
|
end;
|
|
|
|
constructor taicpu.op_sym_ofs(op: tasmop; _op1: tasmsymbol; _op1ofs: longint);
|
|
begin
|
|
inherited Create(op);
|
|
ops := 1;
|
|
loadsymbol(0, _op1, _op1ofs);
|
|
end;
|
|
|
|
|
|
function taicpu.is_same_reg_move(regtype: Tregistertype): boolean;
|
|
begin
|
|
Result := (
|
|
((opcode = A_MOVE) and (regtype = R_INTREGISTER)) or
|
|
((regtype = R_FPUREGISTER) and (opcode in [A_MOV_S, A_MOV_D]))
|
|
) and
|
|
(oper[0]^.reg = oper[1]^.reg);
|
|
end;
|
|
|
|
|
|
function taicpu.is_macro: boolean;
|
|
begin
|
|
result :=
|
|
{ 'seq', 'sge', 'sgeu', 'sgt', 'sgtu', 'sle', 'sleu', 'sne', }
|
|
(opcode=A_SEQ) or (opcode=A_SGE) or (opcode=A_SGEU) or (opcode=A_SGT) or
|
|
(opcode=A_SGTU) or (opcode=A_SLE) or (opcode=A_SLEU) or (opcode=A_SNE)
|
|
{ JAL is not here! See comments in TCGMIPS.a_call_name. }
|
|
or (opcode=A_LA) or ((opcode=A_BC) and
|
|
not (condition in [C_EQ,C_NE,C_GTZ,C_GEZ,C_LTZ,C_LEZ,C_COP1TRUE,C_COP1FALSE])) {or (op=A_JAL)}
|
|
or (opcode=A_REM) or (opcode=A_REMU)
|
|
{ DIV and DIVU are normally macros, but use $zero as first arg to generate a CPU instruction. }
|
|
or (((opcode=A_DIV) or (opcode=A_DIVU)) and
|
|
((ops<>3) or (oper[0]^.typ<>top_reg) or (oper[0]^.reg<>NR_R0)))
|
|
or (opcode=A_MULO) or (opcode=A_MULOU)
|
|
{ A_LI is only a macro if the immediate is not in thez 16-bit range }
|
|
or (opcode=A_LI);
|
|
end;
|
|
|
|
|
|
function taicpu.spilling_get_operation_type(opnr: longint): topertype;
|
|
type
|
|
op_write_set_type = set of TAsmOp;
|
|
const
|
|
op_write_set: op_write_set_type =
|
|
[A_NEG,
|
|
A_NEGU,
|
|
A_LI,
|
|
A_DLI,
|
|
A_LA,
|
|
A_MOVE,
|
|
A_LB,
|
|
A_LBU,
|
|
A_LH,
|
|
A_LHU,
|
|
A_LW,
|
|
A_LWU,
|
|
A_LWL,
|
|
A_LWR,
|
|
A_LD,
|
|
A_LDL,
|
|
A_LDR,
|
|
A_LL,
|
|
A_LLD,
|
|
A_ADDI,
|
|
A_DADDI,
|
|
A_ADDIU,
|
|
A_DADDIU,
|
|
A_SLTI,
|
|
A_SLTIU,
|
|
A_ANDI,
|
|
A_ORI,
|
|
A_XORI,
|
|
A_LUI,
|
|
A_DNEG,
|
|
A_DNEGU,
|
|
A_ADD,
|
|
A_DADD,
|
|
A_ADDU,
|
|
A_DADDU,
|
|
A_SUB,
|
|
A_DSUB,
|
|
A_SUBU,
|
|
A_DSUBU,
|
|
A_SLT,
|
|
A_SLTU,
|
|
A_AND,
|
|
A_OR,
|
|
A_XOR,
|
|
A_NOR,
|
|
{ We can get into trouble if an instruction can be interpreted as
|
|
macros with different operands. The following commented out ones
|
|
refer to elementary instructions: DIV[U], MULT[U] do not modify
|
|
first operand. Rest are subject to check. }
|
|
A_MUL,
|
|
A_MULO,
|
|
A_MULOU,
|
|
A_DMUL,
|
|
A_DMULO,
|
|
A_DMULOU,
|
|
// A_DIV,
|
|
// A_DIVU,
|
|
A_DDIV,
|
|
A_DDIVU,
|
|
A_REM,
|
|
A_REMU,
|
|
A_DREM,
|
|
A_DREMU,
|
|
// A_MULT,
|
|
A_DMULT,
|
|
// A_MULTU,
|
|
A_DMULTU,
|
|
A_MFHI,
|
|
A_MFLO,
|
|
|
|
A_SLL,
|
|
A_SRL,
|
|
A_SRA,
|
|
A_SLLV,
|
|
A_SRLV,
|
|
A_SRAV,
|
|
A_DSLL,
|
|
A_DSRL,
|
|
A_DSRA,
|
|
A_DSLLV,
|
|
A_DSRLV,
|
|
A_DSRAV,
|
|
A_DSLL32,
|
|
A_DSRL32,
|
|
A_DSRA32,
|
|
A_LWC1,
|
|
A_LDC1,
|
|
|
|
|
|
A_ADD_S,
|
|
A_ADD_D,
|
|
A_SUB_S,
|
|
A_SUB_D,
|
|
A_MUL_S,
|
|
A_MUL_D,
|
|
A_DIV_S,
|
|
A_DIV_D,
|
|
A_ABS_S,
|
|
A_ABS_D,
|
|
A_NEG_S,
|
|
A_NEG_D,
|
|
A_SQRT_S,
|
|
A_SQRT_D,
|
|
A_MOV_S,
|
|
A_MOV_D,
|
|
A_CVT_S_D,
|
|
A_CVT_S_W,
|
|
A_CVT_S_L,
|
|
A_CVT_D_S,
|
|
A_CVT_D_W,
|
|
A_CVT_D_L,
|
|
A_CVT_W_S,
|
|
A_CVT_W_D,
|
|
A_CVT_L_S,
|
|
A_CVT_L_D,
|
|
A_ROUND_W_S,
|
|
A_ROUND_W_D,
|
|
A_ROUND_L_S,
|
|
A_ROUND_L_D,
|
|
A_TRUNC_W_S,
|
|
A_TRUNC_W_D,
|
|
A_TRUNC_L_S,
|
|
A_TRUNC_L_D,
|
|
A_CEIL_W_S,
|
|
A_CEIL_W_D,
|
|
A_CEIL_L_S,
|
|
A_CEIL_L_D,
|
|
A_FLOOR_W_S,
|
|
A_FLOOR_W_D,
|
|
A_FLOOR_L_S,
|
|
A_FLOOR_L_D,
|
|
A_SEQ,
|
|
A_SGE,
|
|
A_SGEU,
|
|
A_SGT,
|
|
A_SGTU,
|
|
A_SLE,
|
|
A_SLEU,
|
|
A_SNE,
|
|
A_EXT,
|
|
A_INS,
|
|
A_MFC0,
|
|
A_SEB,
|
|
A_SEH];
|
|
|
|
begin
|
|
result := operand_read;
|
|
case opcode of
|
|
A_DIV, { these have 3 operands if used as macros }
|
|
A_DIVU:
|
|
if (ops=3) and (opnr=0) then
|
|
result:=operand_write;
|
|
else
|
|
if opcode in op_write_set then
|
|
if opnr = 0 then
|
|
result := operand_write;
|
|
end;
|
|
end;
|
|
|
|
|
|
function spilling_create_load(const ref: treference; r: tregister): taicpu;
|
|
begin
|
|
case getregtype(r) of
|
|
R_INTREGISTER :
|
|
result:=taicpu.op_reg_ref(A_LW,r,ref);
|
|
R_FPUREGISTER :
|
|
begin
|
|
case getsubreg(r) of
|
|
R_SUBFS :
|
|
result:=taicpu.op_reg_ref(A_LWC1,r,ref);
|
|
R_SUBFD :
|
|
result:=taicpu.op_reg_ref(A_LDC1,r,ref);
|
|
else
|
|
internalerror(200401042);
|
|
end;
|
|
end
|
|
else
|
|
internalerror(200401041);
|
|
end;
|
|
end;
|
|
|
|
|
|
function spilling_create_store(r: tregister; const ref: treference): taicpu;
|
|
begin
|
|
case getregtype(r) of
|
|
R_INTREGISTER :
|
|
result:=taicpu.op_reg_ref(A_SW,r,ref);
|
|
R_FPUREGISTER :
|
|
begin
|
|
case getsubreg(r) of
|
|
R_SUBFS :
|
|
result:=taicpu.op_reg_ref(A_SWC1,r,ref);
|
|
R_SUBFD :
|
|
result:=taicpu.op_reg_ref(A_SDC1,r,ref);
|
|
else
|
|
internalerror(200401042);
|
|
end;
|
|
end
|
|
else
|
|
internalerror(200401041);
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure InitAsm;
|
|
begin
|
|
end;
|
|
|
|
|
|
procedure DoneAsm;
|
|
begin
|
|
end;
|
|
|
|
|
|
procedure fixup_jmps(list: TAsmList);
|
|
var
|
|
p,pdelayslot: tai;
|
|
newjmp,newnoop: taicpu;
|
|
labelpositions: TFPList;
|
|
instrpos: ptrint;
|
|
l: tasmlabel;
|
|
again: boolean;
|
|
insai: tai;
|
|
|
|
procedure create_pic_load(ai: taicpu; insloc: tai);
|
|
var
|
|
href: treference;
|
|
newins: taicpu;
|
|
begin
|
|
{ validity of operand has been checked by caller }
|
|
href:=ai.oper[ai.ops-1]^.ref^;
|
|
href.refaddr:=addr_pic;
|
|
href.base:=NR_GP;
|
|
newins:=taicpu.op_reg_ref(A_LW,NR_PIC_FUNC,href);
|
|
newins.fileinfo:=ai.fileinfo;
|
|
list.insertbefore(newins,insloc);
|
|
inc(instrpos,2);
|
|
if (href.symbol.bind=AB_LOCAL) then
|
|
begin
|
|
href.refaddr:=addr_low;
|
|
href.base:=NR_NO;
|
|
newins:=taicpu.op_reg_reg_ref(A_ADDIU,NR_PIC_FUNC,NR_PIC_FUNC,href);
|
|
newins.fileinfo:=ai.fileinfo;
|
|
list.insertbefore(newins,insloc);
|
|
inc(instrpos,2);
|
|
end;
|
|
end;
|
|
|
|
begin
|
|
// MIPS relative branch range is +-32K instructions, i.e +-128 kBytes
|
|
// if certainly not enough instructions to cause an overflow, dont bother
|
|
if (list.count<high(smallint)) then
|
|
exit;
|
|
labelpositions := TFPList.create;
|
|
p := tai(list.first);
|
|
instrpos := 1;
|
|
// record label positions
|
|
while assigned(p) do
|
|
begin
|
|
if p.typ = ait_label then
|
|
begin
|
|
if (tai_label(p).labsym.labelnr >= labelpositions.count) then
|
|
labelpositions.count := tai_label(p).labsym.labelnr * 2;
|
|
labelpositions[tai_label(p).labsym.labelnr] := pointer(instrpos);
|
|
end;
|
|
{ ait_const is for jump tables }
|
|
case p.typ of
|
|
ait_instruction:
|
|
{ probleim here: pseudo-instructions can translate into
|
|
several CPU instructions, possibly depending on assembler options,
|
|
to obe on safe side, let's assume a mean of two. }
|
|
inc(instrpos,2);
|
|
ait_const:
|
|
begin
|
|
if (tai_const(p).consttype<>aitconst_32bit) then
|
|
internalerror(2008052101);
|
|
inc(instrpos);
|
|
end;
|
|
end;
|
|
p := tai(p.next);
|
|
end;
|
|
|
|
{ If the number of instructions is below limit, we can't overflow either }
|
|
if (instrpos<high(smallint)) then
|
|
exit;
|
|
// check and fix distances
|
|
repeat
|
|
again := false;
|
|
p := tai(list.first);
|
|
instrpos := 1;
|
|
while assigned(p) do
|
|
begin
|
|
case p.typ of
|
|
ait_label:
|
|
// update labelposition in case it changed due to insertion
|
|
// of jumps
|
|
begin
|
|
// can happen because of newly inserted labels
|
|
if (tai_label(p).labsym.labelnr > labelpositions.count) then
|
|
labelpositions.count := tai_label(p).labsym.labelnr * 2;
|
|
labelpositions[tai_label(p).labsym.labelnr] := pointer(instrpos);
|
|
end;
|
|
ait_instruction:
|
|
begin
|
|
inc(instrpos,2);
|
|
case taicpu(p).opcode of
|
|
A_BA,A_BC:
|
|
if (taicpu(p).ops>0) and (taicpu(p).oper[taicpu(p).ops-1]^.typ=top_ref) and
|
|
assigned(taicpu(p).oper[taicpu(p).ops-1]^.ref^.symbol) and
|
|
(taicpu(p).oper[taicpu(p).ops-1]^.ref^.symbol is tasmlabel) and
|
|
(labelpositions[tasmlabel(taicpu(p).oper[taicpu(p).ops-1]^.ref^.symbol).labelnr] <> NIL) and
|
|
{$push}
|
|
{$q-}
|
|
(ptruint(abs(ptrint(labelpositions[tasmlabel(taicpu(p).oper[taicpu(p).ops-1]^.ref^.symbol).labelnr]-instrpos)) - low(smallint)) > ptruint((high(smallint) - low(smallint)))) then
|
|
{$pop}
|
|
begin
|
|
if (taicpu(p).opcode=A_BC) then
|
|
begin
|
|
{ we're adding a new label together with the only branch to it;
|
|
providing exact label position is not necessary }
|
|
current_asmdata.getjumplabel(l);
|
|
pdelayslot:=tai(p.next);
|
|
{ We need to insert the new instruction after the delay slot instruction ! }
|
|
while assigned(pdelayslot) and (pdelayslot.typ<>ait_instruction) do
|
|
pdelayslot:=tai(pdelayslot.next);
|
|
|
|
insai:=tai_label.create(l);
|
|
list.insertafter(insai,pdelayslot);
|
|
// add a new unconditional jump between this jump and the label
|
|
list.insertbefore(tai_comment.create(strpnew('fixup_jmps, A_BXX changed into A_BNOTXX label;A_J;label:')),p);
|
|
if (cs_create_pic in current_settings.moduleswitches) then
|
|
begin
|
|
create_pic_load(taicpu(p),insai);
|
|
newjmp:=taicpu.op_reg(A_JR,NR_PIC_FUNC);
|
|
end
|
|
else
|
|
begin
|
|
newjmp:=taicpu.op_sym(A_J,taicpu(p).oper[2]^.ref^.symbol);
|
|
newjmp.is_jmp := true;
|
|
end;
|
|
newjmp.fileinfo:=taicpu(p).fileinfo;
|
|
list.insertbefore(newjmp,insai);
|
|
inc(instrpos,2);
|
|
|
|
{ Add a delay slot for new A_J instruction }
|
|
newnoop:=taicpu.op_none(A_NOP);
|
|
newnoop.fileinfo := taicpu(p).fileinfo;
|
|
list.insertbefore(newnoop,insai);
|
|
inc(instrpos,2);
|
|
// change the conditional jump to point to the newly inserted label
|
|
tasmlabel(taicpu(p).oper[taicpu(p).ops-1]^.ref^.symbol).decrefs;
|
|
taicpu(p).oper[taicpu(p).ops-1]^.ref^.symbol := l;
|
|
l.increfs;
|
|
// and invert its condition code
|
|
taicpu(p).condition := inverse_cond(taicpu(p).condition);
|
|
{ skip inserted stuff and continue processing from 'pdelayslot' }
|
|
p:=pdelayslot;
|
|
again:=true;
|
|
end
|
|
else // opcode=A_BA
|
|
begin
|
|
if (cs_create_pic in current_settings.moduleswitches) then
|
|
begin
|
|
list.insertbefore(tai_comment.create(strpnew('fixup_jmps, A_BA changed into PIC sequence')),p);
|
|
create_pic_load(taicpu(p),p);
|
|
taicpu(p).opcode:=A_JR;
|
|
taicpu(p).loadreg(0,NR_PIC_FUNC);
|
|
again:=true;
|
|
{ inserted stuff before 'p', continue processing from 'p' on }
|
|
end
|
|
else
|
|
begin
|
|
list.insertbefore(tai_comment.create(strpnew('fixup_jmps, A_BA changed into A_J')),p);
|
|
taicpu(p).opcode:=A_J;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
ait_const:
|
|
inc(instrpos);
|
|
end;
|
|
p := tai(p.next);
|
|
end;
|
|
until not again;
|
|
labelpositions.free;
|
|
end;
|
|
|
|
|
|
begin
|
|
cai_cpu := taicpu;
|
|
cai_align := tai_align;
|
|
end.
|