| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728 | {    Copyright (c) 2003 by Florian Klaempfl    Member of the Free Pascal development team    This unit implements the code generator for the ARM    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 cgcpu;{$i fpcdefs.inc}  interface    uses       globtype,symtype,symdef,       cgbase,cgutils,cgobj,       aasmbase,aasmcpu,aasmtai,       parabase,       cpubase,cpuinfo,node,cg64f32,rgcpu;    type      tcgarm = class(tcg)        { true, if the next arithmetic operation should modify the flags }        cgsetflags : boolean;        procedure init_register_allocators;override;        procedure done_register_allocators;override;        procedure a_param_const(list : taasmoutput;size : tcgsize;a : aint;const paraloc : TCGPara);override;        procedure a_param_ref(list : taasmoutput;size : tcgsize;const r : treference;const paraloc : TCGPara);override;        procedure a_paramaddr_ref(list : taasmoutput;const r : treference;const paraloc : TCGPara);override;        procedure a_call_name(list : taasmoutput;const s : string);override;        procedure a_call_reg(list : taasmoutput;reg: tregister); override;        procedure a_op_const_reg(list : taasmoutput; Op: TOpCG; size: TCGSize; a: aint; reg: TRegister); override;        procedure a_op_reg_reg(list : taasmoutput; Op: TOpCG; size: TCGSize; src, dst: TRegister); override;        procedure a_op_const_reg_reg(list: taasmoutput; op: TOpCg;          size: tcgsize; a: aint; src, dst: tregister); override;        procedure a_op_reg_reg_reg(list: taasmoutput; op: TOpCg;          size: tcgsize; src1, src2, dst: tregister); override;        procedure a_op_const_reg_reg_checkoverflow(list: taasmoutput; op: TOpCg; size: tcgsize; a: aint; src, dst: tregister;setflags : boolean;var ovloc : tlocation);override;        procedure a_op_reg_reg_reg_checkoverflow(list: taasmoutput; op: TOpCg; size: tcgsize; src1, src2, dst: tregister;setflags : boolean;var ovloc : tlocation);override;        { move instructions }        procedure a_load_const_reg(list : taasmoutput; size: tcgsize; a : aint;reg : tregister);override;        procedure a_load_reg_ref(list : taasmoutput; fromsize, tosize: tcgsize; reg : tregister;const ref : treference);override;        procedure a_load_ref_reg(list : taasmoutput; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister);override;        procedure a_load_reg_reg(list : taasmoutput; fromsize, tosize : tcgsize;reg1,reg2 : tregister);override;        { fpu move instructions }        procedure a_loadfpu_reg_reg(list: taasmoutput; size: tcgsize; reg1, reg2: tregister); override;        procedure a_loadfpu_ref_reg(list: taasmoutput; size: tcgsize; const ref: treference; reg: tregister); override;        procedure a_loadfpu_reg_ref(list: taasmoutput; size: tcgsize; reg: tregister; const ref: treference); override;        procedure a_paramfpu_ref(list : taasmoutput;size : tcgsize;const ref : treference;const paraloc : TCGPara);override;        {  comparison operations }        procedure a_cmp_const_reg_label(list : taasmoutput;size : tcgsize;cmp_op : topcmp;a : aint;reg : tregister;          l : tasmlabel);override;        procedure a_cmp_reg_reg_label(list : taasmoutput;size : tcgsize;cmp_op : topcmp;reg1,reg2 : tregister;l : tasmlabel); override;        procedure a_jmp_name(list : taasmoutput;const s : string); override;        procedure a_jmp_always(list : taasmoutput;l: tasmlabel); override;        procedure a_jmp_flags(list : taasmoutput;const f : TResFlags;l: tasmlabel); override;        procedure g_flags2reg(list: taasmoutput; size: TCgSize; const f: TResFlags; reg: TRegister); override;        procedure g_proc_entry(list : taasmoutput;localsize : longint;nostackframe:boolean);override;        procedure g_proc_exit(list : taasmoutput;parasize : longint;nostackframe:boolean); override;        procedure a_loadaddr_ref_reg(list : taasmoutput;const ref : treference;r : tregister);override;        procedure g_concatcopy(list : taasmoutput;const source,dest : treference;len : aint);override;        procedure g_concatcopy_unaligned(list : taasmoutput;const source,dest : treference;len : aint);override;        procedure g_concatcopy_move(list : taasmoutput;const source,dest : treference;len : aint);        procedure g_concatcopy_internal(list : taasmoutput;const source,dest : treference;len : aint;aligned : boolean);        procedure g_overflowcheck(list: taasmoutput; const l: tlocation; def: tdef); override;        procedure g_overflowCheck_loc(List:TAasmOutput;const Loc:TLocation;def:TDef;ovloc : tlocation);override;        procedure g_save_standard_registers(list : taasmoutput);override;        procedure g_restore_standard_registers(list : taasmoutput);override;        procedure a_jmp_cond(list : taasmoutput;cond : TOpCmp;l: tasmlabel);        procedure fixref(list : taasmoutput;var ref : treference);        procedure handle_load_store(list:taasmoutput;op: tasmop;oppostfix : toppostfix;reg:tregister;ref: treference);        procedure g_intf_wrapper(list: taasmoutput; procdef: tprocdef; const labelname: string; ioffset: longint);override;      end;      tcg64farm = class(tcg64f32)        procedure a_op64_reg_reg(list : taasmoutput;op:TOpCG;size : tcgsize;regsrc,regdst : tregister64);override;        procedure a_op64_const_reg(list : taasmoutput;op:TOpCG;size : tcgsize;value : int64;reg : tregister64);override;        procedure a_op64_const_reg_reg(list: taasmoutput;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64);override;        procedure a_op64_reg_reg_reg(list: taasmoutput;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64);override;        procedure a_op64_const_reg_reg_checkoverflow(list: taasmoutput;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64;setflags : boolean;var ovloc : tlocation);override;        procedure a_op64_reg_reg_reg_checkoverflow(list: taasmoutput;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64;setflags : boolean;var ovloc : tlocation);override;      end;    const      OpCmp2AsmCond : Array[topcmp] of TAsmCond = (C_NONE,C_EQ,C_GT,                           C_LT,C_GE,C_LE,C_NE,C_LS,C_CC,C_CS,C_HI);    function is_shifter_const(d : aint;var imm_shift : byte) : boolean;    function get_fpu_postfix(def : tdef) : toppostfix;  implementation    uses       globals,verbose,systems,cutils,       fmodule,       symconst,symsym,       tgobj,       procinfo,cpupi,       paramgr;    function get_fpu_postfix(def : tdef) : toppostfix;      begin        if def.deftype=floatdef then          begin            case tfloatdef(def).typ of              s32real:                result:=PF_S;              s64real:                result:=PF_D;              s80real:                result:=PF_E;              else                internalerror(200401272);            end;          end        else          internalerror(200401271);      end;    procedure tcgarm.init_register_allocators;      begin        inherited init_register_allocators;        { currently, we save R14 always, so we can use it }        rg[R_INTREGISTER]:=trgintcpu.create(R_INTREGISTER,R_SUBWHOLE,            [RS_R0,RS_R1,RS_R2,RS_R3,RS_R4,RS_R5,RS_R6,RS_R7,RS_R8,             RS_R9,RS_R10,RS_R12,RS_R14],first_int_imreg,[]);        rg[R_FPUREGISTER]:=trgcpu.create(R_FPUREGISTER,R_SUBNONE,            [RS_F0,RS_F1,RS_F2,RS_F3,RS_F4,RS_F5,RS_F6,RS_F7],first_fpu_imreg,[]);        rg[R_MMREGISTER]:=trgcpu.create(R_MMREGISTER,R_SUBNONE,            [RS_S0,RS_S1,RS_R2,RS_R3,RS_R4,RS_S31],first_mm_imreg,[]);      end;    procedure tcgarm.done_register_allocators;      begin        rg[R_INTREGISTER].free;        rg[R_FPUREGISTER].free;        rg[R_MMREGISTER].free;        inherited done_register_allocators;      end;    procedure tcgarm.a_param_const(list : taasmoutput;size : tcgsize;a : aint;const paraloc : TCGPara);      var        ref: treference;      begin        paraloc.check_simple_location;        case paraloc.location^.loc of          LOC_REGISTER,LOC_CREGISTER:            a_load_const_reg(list,size,a,paraloc.location^.register);          LOC_REFERENCE:            begin               reference_reset(ref);               ref.base:=paraloc.location^.reference.index;               ref.offset:=paraloc.location^.reference.offset;               a_load_const_ref(list,size,a,ref);            end;          else            internalerror(2002081101);        end;      end;    procedure tcgarm.a_param_ref(list : taasmoutput;size : tcgsize;const r : treference;const paraloc : TCGPara);      var        ref: treference;        tmpreg: tregister;      begin        paraloc.check_simple_location;        case paraloc.location^.loc of          LOC_REGISTER,LOC_CREGISTER:            a_load_ref_reg(list,size,size,r,paraloc.location^.register);          LOC_REFERENCE:            begin               reference_reset(ref);               ref.base:=paraloc.location^.reference.index;               ref.offset:=paraloc.location^.reference.offset;               tmpreg := getintregister(list,size);               a_load_ref_reg(list,size,size,r,tmpreg);               a_load_reg_ref(list,size,size,tmpreg,ref);            end;          LOC_FPUREGISTER,LOC_CFPUREGISTER:            case size of               OS_F32, OS_F64:                 a_loadfpu_ref_reg(list,size,r,paraloc.location^.register);               else                 internalerror(2002072801);            end;          else            internalerror(2002081103);        end;      end;    procedure tcgarm.a_paramaddr_ref(list : taasmoutput;const r : treference;const paraloc : TCGPara);      var        ref: treference;        tmpreg: tregister;      begin        paraloc.check_simple_location;         case paraloc.location^.loc of            LOC_REGISTER,LOC_CREGISTER:              a_loadaddr_ref_reg(list,r,paraloc.location^.register);            LOC_REFERENCE:              begin                reference_reset(ref);                ref.base := paraloc.location^.reference.index;                ref.offset := paraloc.location^.reference.offset;                tmpreg := getintregister(list,OS_ADDR);                a_loadaddr_ref_reg(list,r,tmpreg);                a_load_reg_ref(list,OS_ADDR,OS_ADDR,tmpreg,ref);              end;            else              internalerror(2002080701);         end;      end;    procedure tcgarm.a_call_name(list : taasmoutput;const s : string);      begin        list.concat(taicpu.op_sym(A_BL,objectlibrary.newasmsymbol(s,AB_EXTERNAL,AT_FUNCTION)));{        the compiler does not properly set this flag anymore in pass 1, and        for now we only need it after pass 2 (I hope) (JM)          if not(pi_do_call in current_procinfo.flags) then            internalerror(2003060703);}        include(current_procinfo.flags,pi_do_call);      end;    procedure tcgarm.a_call_reg(list : taasmoutput;reg: tregister);      var         r : tregister;      begin        list.concat(taicpu.op_reg_reg(A_MOV,NR_R14,NR_PC));        list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,reg));{        the compiler does not properly set this flag anymore in pass 1, and        for now we only need it after pass 2 (I hope) (JM)          if not(pi_do_call in current_procinfo.flags) then            internalerror(2003060703);}        include(current_procinfo.flags,pi_do_call);      end;     procedure tcgarm.a_op_const_reg(list : taasmoutput; Op: TOpCG; size: TCGSize; a: aint; reg: TRegister);       begin          a_op_const_reg_reg(list,op,size,a,reg,reg);       end;     procedure tcgarm.a_op_reg_reg(list : taasmoutput; Op: TOpCG; size: TCGSize; src, dst: TRegister);       begin         case op of           OP_NEG:             list.concat(taicpu.op_reg_reg_const(A_RSB,dst,src,0));           OP_NOT:             begin               list.concat(taicpu.op_reg_reg(A_MVN,dst,src));               case size of                 OS_8 :                   a_op_const_reg_reg(list,OP_AND,OS_INT,$ff,dst,dst);                 OS_16 :                   a_op_const_reg_reg(list,OP_AND,OS_INT,$ffff,dst,dst);               end;             end           else             a_op_reg_reg_reg(list,op,OS_32,src,dst,dst);         end;       end;    const      op_reg_reg_opcg2asmop: array[TOpCG] of tasmop =        (A_NONE,A_ADD,A_AND,A_NONE,A_NONE,A_MUL,A_MUL,A_NONE,A_NONE,A_ORR,         A_NONE,A_NONE,A_NONE,A_SUB,A_EOR);    procedure tcgarm.a_op_const_reg_reg(list: taasmoutput; op: TOpCg;      size: tcgsize; a: aint; src, dst: tregister);      var        ovloc : tlocation;      begin        a_op_const_reg_reg_checkoverflow(list,op,size,a,src,dst,false,ovloc);      end;    procedure tcgarm.a_op_reg_reg_reg(list: taasmoutput; op: TOpCg;      size: tcgsize; src1, src2, dst: tregister);      var        ovloc : tlocation;      begin        a_op_reg_reg_reg_checkoverflow(list,op,size,src1,src2,dst,false,ovloc);      end;    procedure tcgarm.a_op_const_reg_reg_checkoverflow(list: taasmoutput; op: TOpCg; size: tcgsize; a: aint; src, dst: tregister;setflags : boolean;var ovloc : tlocation);      var        shift : byte;        tmpreg : tregister;        so : tshifterop;        l1 : longint;      begin        ovloc.loc:=LOC_VOID;        if is_shifter_const(-a,shift) then          case op of            OP_ADD:              begin                op:=OP_SUB;                a:=dword(-a);              end;            OP_SUB:              begin                op:=OP_ADD;                a:=dword(-a);              end          end;        if is_shifter_const(a,shift) and not(op in [OP_IMUL,OP_MUL]) then          case op of            OP_NEG,OP_NOT,            OP_DIV,OP_IDIV:              internalerror(200308281);            OP_SHL:              begin                if a>32 then                  internalerror(200308291);                if a<>0 then                  begin                    shifterop_reset(so);                    so.shiftmode:=SM_LSL;                    so.shiftimm:=a;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));                  end                else                 list.concat(taicpu.op_reg_reg(A_MOV,dst,src));              end;            OP_SHR:              begin                if a>32 then                  internalerror(200308292);                shifterop_reset(so);                if a<>0 then                  begin                    so.shiftmode:=SM_LSR;                    so.shiftimm:=a;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));                  end                else                 list.concat(taicpu.op_reg_reg(A_MOV,dst,src));              end;            OP_SAR:              begin                if a>32 then                  internalerror(200308291);                if a<>0 then                  begin                    shifterop_reset(so);                    so.shiftmode:=SM_ASR;                    so.shiftimm:=a;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));                  end                else                 list.concat(taicpu.op_reg_reg(A_MOV,dst,src));              end;            else              list.concat(setoppostfix(                  taicpu.op_reg_reg_const(op_reg_reg_opcg2asmop[op],dst,src,a),toppostfix(ord(cgsetflags or setflags)*ord(PF_S))              ));              if (cgsetflags or setflags) and (size in [OS_8,OS_16,OS_32]) then                begin                  ovloc.loc:=LOC_FLAGS;                  case op of                    OP_ADD:                      ovloc.resflags:=F_CS;                    OP_SUB:                      ovloc.resflags:=F_CC;                  end;                end;          end        else          begin            { there could be added some more sophisticated optimizations }            if (op in [OP_MUL,OP_IMUL]) and (a=1) then              a_load_reg_reg(list,size,size,src,dst)            else if (op in [OP_MUL,OP_IMUL]) and (a=0) then              a_load_const_reg(list,size,0,dst)            else if (op in [OP_IMUL]) and (a=-1) then              a_op_reg_reg(list,OP_NEG,size,src,dst)            { we do this here instead in the peephole optimizer because              it saves us a register }            else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a,l1) and not(cgsetflags or setflags) then              a_op_const_reg_reg(list,OP_SHL,size,l1,src,dst)            else              begin                tmpreg:=getintregister(list,size);                a_load_const_reg(list,size,a,tmpreg);                a_op_reg_reg_reg_checkoverflow(list,op,size,tmpreg,src,dst,setflags,ovloc);              end;          end;      end;    procedure tcgarm.a_op_reg_reg_reg_checkoverflow(list: taasmoutput; op: TOpCg; size: tcgsize; src1, src2, dst: tregister;setflags : boolean;var ovloc : tlocation);      var        so : tshifterop;        tmpreg,overflowreg : tregister;        asmop : tasmop;      begin        ovloc.loc:=LOC_VOID;        case op of          OP_NEG,OP_NOT,          OP_DIV,OP_IDIV:            internalerror(200308281);          OP_SHL:            begin              shifterop_reset(so);              so.rs:=src1;              so.shiftmode:=SM_LSL;              list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src2,so));            end;          OP_SHR:            begin              shifterop_reset(so);              so.rs:=src1;              so.shiftmode:=SM_LSR;              list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src2,so));            end;          OP_SAR:            begin              shifterop_reset(so);              so.rs:=src1;              so.shiftmode:=SM_ASR;              list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src2,so));            end;          OP_IMUL,          OP_MUL:            begin              if cgsetflags or setflags then                begin                  overflowreg:=getintregister(list,size);                  if op=OP_IMUL then                    asmop:=A_SMULL                  else                    asmop:=A_UMULL;                  { the arm doesn't allow that rd and rm are the same }                  if dst=src2 then                    begin                      if dst<>src1 then                        list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,src1,src2))                      else                        begin                          tmpreg:=getintregister(list,size);                          a_load_reg_reg(list,size,size,src2,dst);                          list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,tmpreg,src1));                        end;                    end                  else                    list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,src2,src1));                  if op=OP_IMUL then                    begin                      shifterop_reset(so);                      so.shiftmode:=SM_ASR;                      so.shiftimm:=31;                      list.concat(taicpu.op_reg_reg_shifterop(A_CMP,overflowreg,dst,so));                    end                  else                    list.concat(taicpu.op_reg_const(A_CMP,overflowreg,0));                   ovloc.loc:=LOC_FLAGS;                   ovloc.resflags:=F_NE;                end              else                begin                  { the arm doesn't allow that rd and rm are the same }                  if dst=src2 then                    begin                      if dst<>src1 then                        list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,src1,src2))                      else                        begin                          tmpreg:=getintregister(list,size);                          a_load_reg_reg(list,size,size,src2,dst);                          list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,tmpreg,src1));                        end;                    end                  else                    list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,src2,src1));                end;            end;          else            list.concat(setoppostfix(                taicpu.op_reg_reg_reg(op_reg_reg_opcg2asmop[op],dst,src2,src1),toppostfix(ord(cgsetflags or setflags)*ord(PF_S))              ));        end;      end;     function rotl(d : dword;b : byte) : dword;       begin          result:=(d shr (32-b)) or (d shl b);       end;     function is_shifter_const(d : aint;var imm_shift : byte) : boolean;       var          i : longint;       begin          for i:=0 to 15 do            begin               if (dword(d) and not(rotl($ff,i*2)))=0 then                 begin                    imm_shift:=i*2;                    result:=true;                    exit;                 end;            end;          result:=false;       end;     procedure tcgarm.a_load_const_reg(list : taasmoutput; size: tcgsize; a : aint;reg : tregister);       var          imm_shift : byte;          l : tasmlabel;          hr : treference;       begin          if not(size in [OS_8,OS_S8,OS_16,OS_S16,OS_32,OS_S32]) then            internalerror(2002090902);          if is_shifter_const(a,imm_shift) then            list.concat(taicpu.op_reg_const(A_MOV,reg,a))          else if is_shifter_const(not(a),imm_shift) then            list.concat(taicpu.op_reg_const(A_MVN,reg,not(a)))          else            begin               reference_reset(hr);               objectlibrary.getlabel(l);               cg.a_label(current_procinfo.aktlocaldata,l);               hr.symboldata:=current_procinfo.aktlocaldata.last;               current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(longint(a)));               hr.symbol:=l;               list.concat(taicpu.op_reg_ref(A_LDR,reg,hr));            end;       end;    procedure tcgarm.handle_load_store(list:taasmoutput;op: tasmop;oppostfix : toppostfix;reg:tregister;ref: treference);      var        tmpreg : tregister;        tmpref : treference;        l : tasmlabel;      begin        tmpreg:=NR_NO;        { Be sure to have a base register }        if (ref.base=NR_NO) then          begin            if ref.shiftmode<>SM_None then              internalerror(200308294);            ref.base:=ref.index;            ref.index:=NR_NO;          end;        { absolute symbols can't be handled directly, we've to store the symbol reference          in the text segment and access it pc relative          For now, we assume that references where base or index equals to PC are already          relative, all other references are assumed to be absolute and thus they need          to be handled extra.          A proper solution would be to change refoptions to a set and store the information          if the symbol is absolute or relative there.        }        if (assigned(ref.symbol) and            not(is_pc(ref.base)) and            not(is_pc(ref.index))           ) or           { [#xxx] isn't a valid address operand }           ((ref.base=NR_NO) and (ref.index=NR_NO)) or           (ref.offset<-4095) or           (ref.offset>4095) or           ((oppostfix in [PF_SB,PF_H,PF_SH]) and            ((ref.offset<-255) or             (ref.offset>255)            )           ) or           ((op in [A_LDF,A_STF]) and            ((ref.offset<-1020) or             (ref.offset>1020) or             { the usual pc relative symbol handling assumes possible offsets of +/- 4095 }             assigned(ref.symbol)            )           ) then          begin            reference_reset(tmpref);            { load symbol }            tmpreg:=getintregister(list,OS_INT);            if assigned(ref.symbol) then              begin                objectlibrary.getlabel(l);                cg.a_label(current_procinfo.aktlocaldata,l);                tmpref.symboldata:=current_procinfo.aktlocaldata.last;                current_procinfo.aktlocaldata.concat(tai_const.create_sym_offset(ref.symbol,ref.offset));                { load consts entry }                tmpref.symbol:=l;                tmpref.base:=NR_R15;                list.concat(taicpu.op_reg_ref(A_LDR,tmpreg,tmpref));              end            else              a_load_const_reg(list,OS_ADDR,ref.offset,tmpreg);            if (ref.base<>NR_NO) then              begin                if ref.index<>NR_NO then                  begin                    list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,tmpreg));                    ref.base:=tmpreg;                  end                else                  begin                    ref.index:=tmpreg;                    ref.shiftimm:=0;                    ref.signindex:=1;                    ref.shiftmode:=SM_None;                  end;              end            else              ref.base:=tmpreg;            ref.offset:=0;            ref.symbol:=nil;          end;        if (ref.base<>NR_NO) and (ref.index<>NR_NO) and (ref.offset<>0) then          begin            if tmpreg<>NR_NO then              a_op_const_reg_reg(list,OP_ADD,OS_ADDR,ref.offset,tmpreg,tmpreg)            else              begin                tmpreg:=getintregister(list,OS_ADDR);                a_op_const_reg_reg(list,OP_ADD,OS_ADDR,ref.offset,ref.base,tmpreg);                ref.base:=tmpreg;              end;            ref.offset:=0;          end;        { floating point operations have only limited references          we expect here, that a base is already set }        if (op in [A_LDF,A_STF]) and (ref.index<>NR_NO) then          begin            if ref.shiftmode<>SM_none then              internalerror(200309121);            if tmpreg<>NR_NO then              begin                if ref.base=tmpreg then                  begin                    if ref.signindex<0 then                      list.concat(taicpu.op_reg_reg_reg(A_SUB,tmpreg,tmpreg,ref.index))                    else                      list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,tmpreg,ref.index));                    ref.index:=NR_NO;                  end                else                  begin                    if ref.index<>tmpreg then                      internalerror(200403161);                    if ref.signindex<0 then                      list.concat(taicpu.op_reg_reg_reg(A_SUB,tmpreg,ref.base,tmpreg))                    else                      list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,tmpreg));                    ref.base:=tmpreg;                    ref.index:=NR_NO;                  end;              end            else              begin                tmpreg:=getintregister(list,OS_ADDR);                list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,ref.index));                ref.base:=tmpreg;                ref.index:=NR_NO;              end;          end;        list.concat(setoppostfix(taicpu.op_reg_ref(op,reg,ref),oppostfix));      end;     procedure tcgarm.a_load_reg_ref(list : taasmoutput; fromsize, tosize: tcgsize; reg : tregister;const ref : treference);       var         oppostfix:toppostfix;       begin         case ToSize of           { signed integer registers }           OS_8,           OS_S8:             oppostfix:=PF_B;           OS_16,           OS_S16:             oppostfix:=PF_H;           OS_32,           OS_S32:             oppostfix:=PF_None;           else             InternalError(200308295);         end;         handle_load_store(list,A_STR,oppostfix,reg,ref);       end;     procedure tcgarm.a_load_ref_reg(list : taasmoutput; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister);       var         oppostfix:toppostfix;       begin         case FromSize of           { signed integer registers }           OS_8:             oppostfix:=PF_B;           OS_S8:             oppostfix:=PF_SB;           OS_16:             oppostfix:=PF_H;           OS_S16:             oppostfix:=PF_SH;           OS_32,           OS_S32:             oppostfix:=PF_None;           else             InternalError(200308291);         end;         handle_load_store(list,A_LDR,oppostfix,reg,ref);       end;     procedure tcgarm.a_load_reg_reg(list : taasmoutput; fromsize, tosize : tcgsize;reg1,reg2 : tregister);       var         instr: taicpu;         so : tshifterop;       begin         shifterop_reset(so);         if (tcgsize2size[tosize] < tcgsize2size[fromsize]) or            (              (tcgsize2size[tosize] = tcgsize2size[fromsize]) and             (tosize <> fromsize) and             not(fromsize in [OS_32,OS_S32])            ) then           begin             case tosize of               OS_8:                 list.concat(taicpu.op_reg_reg_const(A_AND,                   reg2,reg1,$ff));               OS_S8:                 begin                   so.shiftmode:=SM_LSL;                   so.shiftimm:=24;                   list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg1,so));                   so.shiftmode:=SM_ASR;                   so.shiftimm:=24;                   list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg2,so));                 end;               OS_16:                 begin                   so.shiftmode:=SM_LSL;                   so.shiftimm:=16;                   list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg1,so));                   so.shiftmode:=SM_LSR;                   so.shiftimm:=16;                   list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg2,so));                 end;               OS_S16:                 begin                   so.shiftmode:=SM_LSL;                   so.shiftimm:=16;                   list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg1,so));                   so.shiftmode:=SM_ASR;                   so.shiftimm:=16;                   list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg2,so));                 end;               OS_32,OS_S32:                 begin                   instr:=taicpu.op_reg_reg(A_MOV,reg2,reg1);                   list.concat(instr);                   add_move_instruction(instr);                 end;               else internalerror(2002090901);             end;           end         else           begin             if reg1<>reg2 then               begin                 { same size, only a register mov required }                 instr:=taicpu.op_reg_reg(A_MOV,reg2,reg1);                 list.Concat(instr);                 { Notify the register allocator that we have written a move instruction so                   it can try to eliminate it. }                 add_move_instruction(instr);               end;           end;       end;    procedure tcgarm.a_paramfpu_ref(list : taasmoutput;size : tcgsize;const ref : treference;const paraloc : TCGPara);      var         href,href2 : treference;         hloc : pcgparalocation;      begin        href:=ref;        hloc:=paraloc.location;        while assigned(hloc) do          begin            case hloc^.loc of              LOC_FPUREGISTER,LOC_CFPUREGISTER:                a_loadfpu_ref_reg(list,size,ref,hloc^.register);              LOC_REGISTER :                a_load_ref_reg(list,hloc^.size,hloc^.size,href,hloc^.register);              LOC_REFERENCE :                begin                  reference_reset_base(href2,hloc^.reference.index,hloc^.reference.offset);                  a_load_ref_ref(list,hloc^.size,hloc^.size,href,href2);                end;              else                internalerror(200408241);           end;           inc(href.offset,tcgsize2size[hloc^.size]);           hloc:=hloc^.next;         end;      end;     procedure tcgarm.a_loadfpu_reg_reg(list: taasmoutput; size: tcgsize; reg1, reg2: tregister);       begin         list.concat(setoppostfix(taicpu.op_reg_reg(A_MVF,reg2,reg1),cgsize2fpuoppostfix[size]));       end;     procedure tcgarm.a_loadfpu_ref_reg(list: taasmoutput; size: tcgsize; const ref: treference; reg: tregister);       var         oppostfix:toppostfix;       begin         case size of           OS_F32:             oppostfix:=PF_S;           OS_F64:             oppostfix:=PF_D;           OS_F80:             oppostfix:=PF_E;           else             InternalError(200309021);         end;         handle_load_store(list,A_LDF,oppostfix,reg,ref);       end;     procedure tcgarm.a_loadfpu_reg_ref(list: taasmoutput; size: tcgsize; reg: tregister; const ref: treference);       var         oppostfix:toppostfix;       begin         case size of           OS_F32:             oppostfix:=PF_S;           OS_F64:             oppostfix:=PF_D;           OS_F80:             oppostfix:=PF_E;           else             InternalError(200309021);         end;         handle_load_store(list,A_STF,oppostfix,reg,ref);       end;    {  comparison operations }    procedure tcgarm.a_cmp_const_reg_label(list : taasmoutput;size : tcgsize;cmp_op : topcmp;a : aint;reg : tregister;      l : tasmlabel);      var        tmpreg : tregister;        b : byte;      begin        if is_shifter_const(a,b) then          list.concat(taicpu.op_reg_const(A_CMP,reg,a))        { CMN reg,0 and CMN reg,$80000000 are different from CMP reg,$ffffffff          and CMP reg,$7fffffff regarding the flags according to the ARM manual }        else if (a<>$7fffffff) and (a<>-1) and is_shifter_const(-a,b) then          list.concat(taicpu.op_reg_const(A_CMN,reg,-a))        else          begin            tmpreg:=getintregister(list,size);            a_load_const_reg(list,size,a,tmpreg);            list.concat(taicpu.op_reg_reg(A_CMP,reg,tmpreg));          end;        a_jmp_cond(list,cmp_op,l);      end;    procedure tcgarm.a_cmp_reg_reg_label(list : taasmoutput;size : tcgsize;cmp_op : topcmp;reg1,reg2 : tregister;l : tasmlabel);      begin        list.concat(taicpu.op_reg_reg(A_CMP,reg2,reg1));        a_jmp_cond(list,cmp_op,l);      end;    procedure tcgarm.a_jmp_name(list : taasmoutput;const s : string);      begin        list.concat(taicpu.op_sym(A_B,objectlibrary.newasmsymbol(s,AB_EXTERNAL,AT_FUNCTION)));      end;    procedure tcgarm.a_jmp_always(list : taasmoutput;l: tasmlabel);      begin        list.concat(taicpu.op_sym(A_B,objectlibrary.newasmsymbol(l.name,AB_EXTERNAL,AT_FUNCTION)));      end;    procedure tcgarm.a_jmp_flags(list : taasmoutput;const f : TResFlags;l: tasmlabel);      var        ai : taicpu;      begin        ai:=setcondition(taicpu.op_sym(A_B,l),flags_to_cond(f));        ai.is_jmp:=true;        list.concat(ai);      end;    procedure tcgarm.g_flags2reg(list: taasmoutput; size: TCgSize; const f: TResFlags; reg: TRegister);      var        ai : taicpu;      begin        list.concat(setcondition(taicpu.op_reg_const(A_MOV,reg,1),flags_to_cond(f)));        list.concat(setcondition(taicpu.op_reg_const(A_MOV,reg,0),inverse_cond(flags_to_cond(f))));      end;    procedure tcgarm.g_proc_entry(list : taasmoutput;localsize : longint;nostackframe:boolean);      var         ref : treference;         shift : byte;         firstfloatreg,lastfloatreg,         r : byte;      begin        LocalSize:=align(LocalSize,4);        if not(nostackframe) then          begin            firstfloatreg:=RS_NO;            { save floating point registers? }            for r:=RS_F0 to RS_F7 do              if r in rg[R_FPUREGISTER].used_in_proc-paramanager.get_volatile_registers_fpu(pocall_stdcall) then                begin                  if firstfloatreg=RS_NO then                    firstfloatreg:=r;                  lastfloatreg:=r;                end;            a_reg_alloc(list,NR_STACK_POINTER_REG);            a_reg_alloc(list,NR_FRAME_POINTER_REG);            a_reg_alloc(list,NR_R12);            list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_STACK_POINTER_REG));            { save int registers }            reference_reset(ref);            ref.index:=NR_STACK_POINTER_REG;            ref.addressmode:=AM_PREINDEXED;            list.concat(setoppostfix(taicpu.op_ref_regset(A_STM,ref,              rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall)+[RS_R11,RS_R12,RS_R14,RS_R15]),              PF_FD));            list.concat(taicpu.op_reg_reg_const(A_SUB,NR_FRAME_POINTER_REG,NR_R12,4));            { allocate necessary stack size }            { don't use  a_op_const_reg_reg here because we don't allow register allocations              in the entry/exit code }            if not(is_shifter_const(localsize,shift)) then              begin                a_load_const_reg(list,OS_ADDR,LocalSize,NR_R12);                list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R12));                a_reg_dealloc(list,NR_R12);              end            else              begin                a_reg_dealloc(list,NR_R12);                list.concat(taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,LocalSize));              end;            if firstfloatreg<>RS_NO then              begin                reference_reset(ref);                if not(is_shifter_const(-tarmprocinfo(current_procinfo).floatregstart,shift)) then                  begin                    a_load_const_reg(list,OS_ADDR,-tarmprocinfo(current_procinfo).floatregstart,NR_R12);                    list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_R12,NR_FRAME_POINTER_REG,NR_R12));                    ref.base:=NR_R12;                  end                else                  begin                    ref.base:=NR_FRAME_POINTER_REG;                    ref.offset:=tarmprocinfo(current_procinfo).floatregstart;                  end;                list.concat(taicpu.op_reg_const_ref(A_SFM,newreg(R_FPUREGISTER,firstfloatreg,R_SUBWHOLE),                  lastfloatreg-firstfloatreg+1,ref));              end;          end;      end;    procedure tcgarm.g_proc_exit(list : taasmoutput;parasize : longint;nostackframe:boolean);      var         ref : treference;         firstfloatreg,lastfloatreg,         r : byte;         shift : byte;      begin        if not(nostackframe) then          begin            { restore floating point register }            firstfloatreg:=RS_NO;            { save floating point registers? }            for r:=RS_F0 to RS_F7 do              if r in rg[R_FPUREGISTER].used_in_proc-paramanager.get_volatile_registers_fpu(pocall_stdcall) then                begin                  if firstfloatreg=RS_NO then                    firstfloatreg:=r;                  lastfloatreg:=r;                end;            if firstfloatreg<>RS_NO then              begin                reference_reset(ref);                if not(is_shifter_const(-tarmprocinfo(current_procinfo).floatregstart,shift)) then                  begin                    a_load_const_reg(list,OS_ADDR,-tarmprocinfo(current_procinfo).floatregstart,NR_R12);                    list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_R12,NR_FRAME_POINTER_REG,NR_R12));                    ref.base:=NR_R12;                  end                else                  begin                    ref.base:=NR_FRAME_POINTER_REG;                    ref.offset:=tarmprocinfo(current_procinfo).floatregstart;                  end;                list.concat(taicpu.op_reg_const_ref(A_LFM,newreg(R_FPUREGISTER,firstfloatreg,R_SUBWHOLE),                  lastfloatreg-firstfloatreg+1,ref));              end;            if (current_procinfo.framepointer=NR_STACK_POINTER_REG) then              list.concat(taicpu.op_reg_reg(A_MOV,NR_R15,NR_R14))            else              begin                { restore int registers and return }                reference_reset(ref);                ref.index:=NR_FRAME_POINTER_REG;                list.concat(setoppostfix(taicpu.op_ref_regset(A_LDM,ref,rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall)+[RS_R11,RS_R13,RS_R15]),PF_EA));              end;          end        else          list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R14));      end;    procedure tcgarm.a_loadaddr_ref_reg(list : taasmoutput;const ref : treference;r : tregister);      var        b : byte;        tmpref : treference;        instr : taicpu;      begin        if ref.addressmode<>AM_OFFSET then          internalerror(200309071);        tmpref:=ref;        { Be sure to have a base register }        if (tmpref.base=NR_NO) then          begin            if tmpref.shiftmode<>SM_None then              internalerror(200308294);            if tmpref.signindex<0 then              internalerror(200312023);            tmpref.base:=tmpref.index;            tmpref.index:=NR_NO;          end;        if assigned(tmpref.symbol) or           not((is_shifter_const(tmpref.offset,b)) or               (is_shifter_const(-tmpref.offset,b))              ) then          fixref(list,tmpref);        { expect a base here if there is an index }        if (tmpref.base=NR_NO) and (tmpref.index<>NR_NO) then          internalerror(200312022);        if tmpref.index<>NR_NO then          begin            if tmpref.shiftmode<>SM_None then              internalerror(200312021);            if tmpref.signindex<0 then              a_op_reg_reg_reg(list,OP_SUB,OS_ADDR,tmpref.base,tmpref.index,r)            else              a_op_reg_reg_reg(list,OP_ADD,OS_ADDR,tmpref.base,tmpref.index,r);            if tmpref.offset<>0 then              a_op_const_reg_reg(list,OP_ADD,OS_ADDR,tmpref.offset,r,r);          end        else          begin            if tmpref.offset<>0 then              begin                if tmpref.base<>NR_NO then                  a_op_const_reg_reg(list,OP_ADD,OS_ADDR,tmpref.offset,tmpref.base,r)                else                  a_load_const_reg(list,OS_ADDR,tmpref.offset,r);              end            else              begin                instr:=taicpu.op_reg_reg(A_MOV,r,tmpref.base);                list.concat(instr);                add_move_instruction(instr);              end;          end;      end;    procedure tcgarm.fixref(list : taasmoutput;var ref : treference);      var        tmpreg : tregister;        tmpref : treference;        l : tasmlabel;      begin        { absolute symbols can't be handled directly, we've to store the symbol reference          in the text segment and access it pc relative          For now, we assume that references where base or index equals to PC are already          relative, all other references are assumed to be absolute and thus they need          to be handled extra.          A proper solution would be to change refoptions to a set and store the information          if the symbol is absolute or relative there.        }        { create consts entry }        reference_reset(tmpref);        objectlibrary.getlabel(l);        cg.a_label(current_procinfo.aktlocaldata,l);        tmpref.symboldata:=current_procinfo.aktlocaldata.last;        if assigned(ref.symbol) then          current_procinfo.aktlocaldata.concat(tai_const.create_sym_offset(ref.symbol,ref.offset))        else          current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(ref.offset));        { load consts entry }        tmpreg:=getintregister(list,OS_INT);        tmpref.symbol:=l;        tmpref.base:=NR_PC;        list.concat(taicpu.op_reg_ref(A_LDR,tmpreg,tmpref));        if (ref.base<>NR_NO) then          begin            if ref.index<>NR_NO then              begin                list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,tmpreg));                ref.base:=tmpreg;              end            else              begin                ref.index:=tmpreg;                ref.shiftimm:=0;                ref.signindex:=1;                ref.shiftmode:=SM_None;              end;          end        else          ref.base:=tmpreg;        ref.offset:=0;        ref.symbol:=nil;      end;    procedure tcgarm.g_concatcopy_move(list : taasmoutput;const source,dest : treference;len : aint);      var        paraloc1,paraloc2,paraloc3 : TCGPara;      begin        paraloc1.init;        paraloc2.init;        paraloc3.init;        paramanager.getintparaloc(pocall_default,1,paraloc1);        paramanager.getintparaloc(pocall_default,2,paraloc2);        paramanager.getintparaloc(pocall_default,3,paraloc3);        paramanager.allocparaloc(list,paraloc3);        a_param_const(list,OS_INT,len,paraloc3);        paramanager.allocparaloc(list,paraloc2);        a_paramaddr_ref(list,dest,paraloc2);        paramanager.allocparaloc(list,paraloc2);        a_paramaddr_ref(list,source,paraloc1);        paramanager.freeparaloc(list,paraloc3);        paramanager.freeparaloc(list,paraloc2);        paramanager.freeparaloc(list,paraloc1);        alloccpuregisters(list,R_INTREGISTER,paramanager.get_volatile_registers_int(pocall_default));        alloccpuregisters(list,R_FPUREGISTER,paramanager.get_volatile_registers_fpu(pocall_default));        a_call_name(list,'FPC_MOVE');        dealloccpuregisters(list,R_FPUREGISTER,paramanager.get_volatile_registers_fpu(pocall_default));        dealloccpuregisters(list,R_INTREGISTER,paramanager.get_volatile_registers_int(pocall_default));        paraloc3.done;        paraloc2.done;        paraloc1.done;      end;    procedure tcgarm.g_concatcopy_internal(list : taasmoutput;const source,dest : treference;len : aint;aligned : boolean);      var        srcref,dstref:treference;        srcreg,destreg,countreg,r:tregister;        helpsize:aword;        copysize:byte;        cgsize:Tcgsize;      procedure genloop(count : aword;size : byte);        const          size2opsize : array[1..4] of tcgsize = (OS_8,OS_16,OS_NO,OS_32);        var          l : tasmlabel;        begin          objectlibrary.getlabel(l);          a_load_const_reg(list,OS_INT,count,countreg);          cg.a_label(list,l);          srcref.addressmode:=AM_POSTINDEXED;          dstref.addressmode:=AM_POSTINDEXED;          srcref.offset:=size;          dstref.offset:=size;          r:=getintregister(list,size2opsize[size]);          a_load_ref_reg(list,size2opsize[size],size2opsize[size],srcref,r);          list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SUB,countreg,countreg,1),PF_S));          a_load_reg_ref(list,size2opsize[size],size2opsize[size],r,dstref);          list.concat(setcondition(taicpu.op_sym(A_B,l),C_NE));          { keep the registers alive }          list.concat(taicpu.op_reg_reg(A_MOV,countreg,countreg));          list.concat(taicpu.op_reg_reg(A_MOV,srcreg,srcreg));          list.concat(taicpu.op_reg_reg(A_MOV,destreg,destreg));        end;      begin        if len=0 then          exit;        helpsize:=12;        dstref:=dest;        srcref:=source;        if cs_littlesize in aktglobalswitches then          helpsize:=8;        if (len<=helpsize) and aligned then          begin            copysize:=4;            cgsize:=OS_32;            while len<>0 do              begin                if len<2 then                  begin                    copysize:=1;                    cgsize:=OS_8;                  end                else if len<4 then                  begin                    copysize:=2;                    cgsize:=OS_16;                  end;                dec(len,copysize);                r:=getintregister(list,cgsize);                a_load_ref_reg(list,cgsize,cgsize,srcref,r);                a_load_reg_ref(list,cgsize,cgsize,r,dstref);                inc(srcref.offset,copysize);                inc(dstref.offset,copysize);              end;          end        else          begin            destreg:=getintregister(list,OS_ADDR);            a_loadaddr_ref_reg(list,dest,destreg);            reference_reset_base(dstref,destreg,0);            srcreg:=getintregister(list,OS_ADDR);            a_loadaddr_ref_reg(list,source,srcreg);            reference_reset_base(srcref,srcreg,0);            countreg:=getintregister(list,OS_32);//            if cs_littlesize in aktglobalswitches  then              genloop(len,1);{            else              begin                helpsize:=len shr 2;                len:=len and 3;                if helpsize>1 then                  begin                    a_load_const_reg(list,OS_INT,helpsize,countreg);                    list.concat(Taicpu.op_none(A_REP,S_NO));                  end;                if helpsize>0 then                  list.concat(Taicpu.op_none(A_MOVSD,S_NO));                if len>1 then                  begin                    dec(len,2);                    list.concat(Taicpu.op_none(A_MOVSW,S_NO));                  end;                if len=1 then                  list.concat(Taicpu.op_none(A_MOVSB,S_NO));                end;}          end;      end;    procedure tcgarm.g_concatcopy_unaligned(list : taasmoutput;const source,dest : treference;len : aint);      begin        g_concatcopy_internal(list,source,dest,len,false);      end;    procedure tcgarm.g_concatcopy(list : taasmoutput;const source,dest : treference;len : aint);      begin        g_concatcopy_internal(list,source,dest,len,true);      end;    procedure tcgarm.g_overflowCheck(list : taasmoutput;const l : tlocation;def : tdef);      var        ovloc : tlocation;      begin        ovloc.loc:=LOC_VOID;        g_overflowCheck_loc(list,l,def,ovloc);      end;    procedure tcgarm.g_overflowCheck_loc(List:TAasmOutput;const Loc:TLocation;def:TDef;ovloc : tlocation);      var        hl : tasmlabel;        ai:TAiCpu;        hflags : tresflags;      begin        if not(cs_check_overflow in aktlocalswitches) then          exit;        objectlibrary.getlabel(hl);        case ovloc.loc of          LOC_VOID:            begin              ai:=taicpu.op_sym(A_B,hl);              ai.is_jmp:=true;              if not((def.deftype=pointerdef) or                    ((def.deftype=orddef) and                     (torddef(def).typ in [u64bit,u16bit,u32bit,u8bit,uchar,bool8bit,bool16bit,bool32bit]))) then                 ai.SetCondition(C_VC)              else                 ai.SetCondition(C_CC);              list.concat(ai);            end;          LOC_FLAGS:            begin              hflags:=ovloc.resflags;              inverse_flags(hflags);              cg.a_jmp_flags(list,hflags,hl);            end;          else            internalerror(200409281);        end;        a_call_name(list,'FPC_OVERFLOW');        a_label(list,hl);      end;    procedure tcgarm.g_save_standard_registers(list : taasmoutput);      begin        { this work is done in g_proc_entry }      end;    procedure tcgarm.g_restore_standard_registers(list : taasmoutput);      begin        { this work is done in g_proc_exit }      end;    procedure tcgarm.a_jmp_cond(list : taasmoutput;cond : TOpCmp;l: tasmlabel);      var        ai : taicpu;      begin        ai:=Taicpu.Op_sym(A_B,l);        ai.SetCondition(OpCmp2AsmCond[cond]);        ai.is_jmp:=true;        list.concat(ai);      end;    procedure tcgarm.g_intf_wrapper(list: taasmoutput; procdef: tprocdef; const labelname: string; ioffset: longint);      procedure loadvmttor12;        var          href : treference;        begin          reference_reset_base(href,NR_R0,0);          cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,href,NR_R12);        end;      procedure op_onr12methodaddr;        var          href : treference;        begin          if (procdef.extnumber=$ffff) then            Internalerror(200006139);          { call/jmp  vmtoffs(%eax) ; method offs }          reference_reset_base(href,NR_R12,procdef._class.vmtmethodoffset(procdef.extnumber));          cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,href,NR_R12);          list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R12));        end;      var        lab : tasmsymbol;        make_global : boolean;        href : treference;      begin        if not(procdef.proctypeoption in [potype_function,potype_procedure]) then          Internalerror(200006137);        if not assigned(procdef._class) or           (procdef.procoptions*[po_classmethod, po_staticmethod,             po_methodpointer, po_interrupt, po_iocheck]<>[]) then          Internalerror(200006138);        if procdef.owner.symtabletype<>objectsymtable then          Internalerror(200109191);        make_global:=false;        if (not current_module.is_unit) or           (cs_create_smart in aktmoduleswitches) or           (procdef.owner.defowner.owner.symtabletype=globalsymtable) then          make_global:=true;        if make_global then          list.concat(Tai_symbol.Createname_global(labelname,AT_FUNCTION,0))        else          list.concat(Tai_symbol.Createname(labelname,AT_FUNCTION,0));        { set param1 interface to self  }        g_adjust_self_value(list,procdef,ioffset);        { case 4 }        if po_virtualmethod in procdef.procoptions then          begin            loadvmttor12;            op_onr12methodaddr;          end        { case 0 }        else          list.concat(taicpu.op_sym(A_B,objectlibrary.newasmsymbol(procdef.mangledname,AB_EXTERNAL,AT_FUNCTION)));        list.concat(Tai_symbol_end.Createname(labelname));      end;    procedure tcg64farm.a_op64_reg_reg(list : taasmoutput;op:TOpCG;size : tcgsize;regsrc,regdst : tregister64);      var        tmpreg : tregister;      begin        case op of          OP_NEG:            begin              list.concat(setoppostfix(taicpu.op_reg_reg_const(A_RSB,regdst.reglo,regsrc.reglo,0),PF_S));              list.concat(taicpu.op_reg_reg_const(A_RSC,regdst.reghi,regsrc.reghi,0));            end;          OP_NOT:            begin              cg.a_op_reg_reg(list,OP_NOT,OS_INT,regsrc.reglo,regdst.reglo);              cg.a_op_reg_reg(list,OP_NOT,OS_INT,regsrc.reghi,regdst.reghi);            end;          else            a_op64_reg_reg_reg(list,op,size,regsrc,regdst,regdst);        end;      end;    procedure tcg64farm.a_op64_const_reg(list : taasmoutput;op:TOpCG;size : tcgsize;value : int64;reg : tregister64);      begin        a_op64_const_reg_reg(list,op,size,value,reg,reg);      end;    procedure tcg64farm.a_op64_const_reg_reg(list: taasmoutput;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64);      var        ovloc : tlocation;      begin        a_op64_const_reg_reg_checkoverflow(list,op,size,value,regsrc,regdst,false,ovloc);      end;    procedure tcg64farm.a_op64_reg_reg_reg(list: taasmoutput;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64);      var        ovloc : tlocation;      begin        a_op64_reg_reg_reg_checkoverflow(list,op,size,regsrc1,regsrc2,regdst,false,ovloc);      end;    procedure tcg64farm.a_op64_const_reg_reg_checkoverflow(list: taasmoutput;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64;setflags : boolean;var ovloc : tlocation);      var        tmpreg : tregister;        b : byte;      begin        ovloc.loc:=LOC_VOID;        case op of          OP_NEG,          OP_NOT :            internalerror(200306017);        end;        if (setflags or tcgarm(cg).cgsetflags) and (op in [OP_ADD,OP_SUB]) then          begin            case op of              OP_ADD:                begin                  if is_shifter_const(lo(value),b) then                    list.concat(setoppostfix(taicpu.op_reg_reg_const(A_ADD,regdst.reglo,regsrc.reglo,lo(value)),PF_S))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,lo(value),tmpreg);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc.reglo,tmpreg),PF_S));                    end;                  if is_shifter_const(hi(value),b) then                    list.concat(setoppostfix(taicpu.op_reg_reg_const(A_ADC,regdst.reghi,regsrc.reghi,hi(value)),PF_S))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc.reghi,tmpreg),PF_S));                    end;                end;              OP_SUB:                begin                  if is_shifter_const(lo(value),b) then                    list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SUB,regdst.reglo,regsrc.reglo,lo(value)),PF_S))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,lo(value),tmpreg);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc.reglo,tmpreg),PF_S));                    end;                  if is_shifter_const(hi(value),b) then                    list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SBC,regdst.reghi,regsrc.reghi,hi(value)),PF_S))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc.reghi,tmpreg),PF_S));                    end;                end;              else                internalerror(200502131);            end;            if size=OS_64 then              begin                { the arm has an weired opinion how flags for SUB/ADD are handled }                ovloc.loc:=LOC_FLAGS;                case op of                  OP_ADD:                    ovloc.resflags:=F_CS;                  OP_SUB:                    ovloc.resflags:=F_CC;                end;              end;          end        else          begin            case op of              OP_AND,OP_OR,OP_XOR:                begin                  cg.a_op_const_reg_reg(list,op,OS_32,lo(value),regsrc.reglo,regdst.reglo);                  cg.a_op_const_reg_reg(list,op,OS_32,hi(value),regsrc.reghi,regdst.reghi);                end;              OP_ADD:                begin                  if is_shifter_const(lo(value),b) then                    list.concat(setoppostfix(taicpu.op_reg_reg_const(A_ADD,regdst.reglo,regsrc.reglo,lo(value)),PF_S))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,lo(value),tmpreg);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc.reglo,tmpreg),PF_S));                    end;                  if is_shifter_const(hi(value),b) then                    list.concat(taicpu.op_reg_reg_const(A_ADC,regdst.reghi,regsrc.reghi,hi(value)))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);                      list.concat(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc.reghi,tmpreg));                    end;                end;              OP_SUB:                begin                  if is_shifter_const(lo(value),b) then                    list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SUB,regdst.reglo,regsrc.reglo,lo(value)),PF_S))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,lo(value),tmpreg);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc.reglo,tmpreg),PF_S));                    end;                  if is_shifter_const(hi(value),b) then                    list.concat(taicpu.op_reg_reg_const(A_SBC,regdst.reghi,regsrc.reghi,hi(value)))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);                      list.concat(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc.reghi,tmpreg));                    end;                end;            else              internalerror(2003083101);          end;        end;      end;    procedure tcg64farm.a_op64_reg_reg_reg_checkoverflow(list: taasmoutput;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64;setflags : boolean;var ovloc : tlocation);      var        op1,op2:TAsmOp;      begin        ovloc.loc:=LOC_VOID;        case op of          OP_NEG,          OP_NOT :            internalerror(200306017);        end;        if (setflags or tcgarm(cg).cgsetflags) and (op in [OP_ADD,OP_SUB]) then          begin            case op of              OP_ADD:                begin                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc1.reglo,regsrc2.reglo),PF_S));                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc1.reghi,regsrc2.reghi),PF_S));                end;              OP_SUB:                begin                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc2.reglo,regsrc1.reglo),PF_S));                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc2.reghi,regsrc1.reghi),PF_S));                end;              else                internalerror(2003083101);            end;            if size=OS_64 then              begin                { the arm has an weired opinion how flags for SUB/ADD are handled }                ovloc.loc:=LOC_FLAGS;                case op of                  OP_ADD:                    ovloc.resflags:=F_CC;                  OP_SUB:                    ovloc.resflags:=F_CS;                end;              end;          end        else          begin            case op of              OP_AND,OP_OR,OP_XOR:                begin                  cg.a_op_reg_reg_reg(list,op,OS_32,regsrc1.reglo,regsrc2.reglo,regdst.reglo);                  cg.a_op_reg_reg_reg(list,op,OS_32,regsrc1.reghi,regsrc2.reghi,regdst.reghi);                end;              OP_ADD:                begin                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc1.reglo,regsrc2.reglo),PF_S));                  list.concat(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc1.reghi,regsrc2.reghi));                end;              OP_SUB:                begin                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc2.reglo,regsrc1.reglo),PF_S));                  list.concat(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc2.reghi,regsrc1.reghi));                end;              else                internalerror(2003083101);            end;          end;      end;begin  cg:=tcgarm.create;  cg64:=tcg64farm.create;end.
 |