| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682 |         {    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}interfaceuses  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 a macro in pic code mode }          or ((opcode=A_JAL) and (cs_create_pic in current_settings.moduleswitches))          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 (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(2004010418);              end;            end          else            internalerror(2004010408);        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(2004010419);              end;            end          else            internalerror(2004010409);        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;          else            ;        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;                  else                    ;                end;              end;            ait_const:              inc(instrpos);            else              ;          end;          p := tai(p.next);        end;     until not again;    labelpositions.free;  end;begin  cai_cpu   := taicpu;  cai_align := tai_align;end.
 |