mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-11-04 16:59:45 +01: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.
 |