123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872 |
- {
- Copyright (c) 1998-2020 by Jonas Maebe and Florian Klaempfl, members of the Free Pascal
- Development Team
- This unit implements an ARM optimizer object used commonly for ARM and AAarch64
- 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 aoptarm;
- {$i fpcdefs.inc}
- { $define DEBUG_PREREGSCHEDULER}
- {$ifdef EXTDEBUG}
- {$define DEBUG_AOPTCPU}
- {$endif EXTDEBUG}
- Interface
- uses
- cgbase, cgutils, globtype, cpubase, aasmtai, aasmcpu,aopt, aoptobj;
- Type
- { while ARM and AAarch64 look not very similar at a first glance,
- several optimizations can be shared between both }
- TARMAsmOptimizer = class(TAsmOptimizer)
- procedure DebugMsg(const s : string; p : tai);
- function RemoveSuperfluousMove(const p: tai; movp: tai; const optimizer: string): boolean;
- function RedundantMovProcess(var p: tai; var hp1: tai): boolean;
- function GetNextInstructionUsingReg(Current: tai; out Next: tai; const reg: TRegister): Boolean;
- {$ifdef AARCH64}
- function USxtOp2Op(var p, hp1: tai; shiftmode: tshiftmode): Boolean;
- {$endif AARCH64}
- function OptPreSBFXUBFX(var p: tai): Boolean;
- function OptPass1UXTB(var p: tai): Boolean;
- function OptPass1UXTH(var p: tai): Boolean;
- function OptPass1SXTB(var p: tai): Boolean;
- function OptPass1SXTH(var p: tai): Boolean;
- function OptPass1LDR(var p: tai): Boolean; virtual;
- function OptPass1STR(var p: tai): Boolean; virtual;
- function OptPass1And(var p: tai): Boolean; virtual;
- function OptPass2Bitwise(var p: tai): Boolean;
- function OptPass2TST(var p: tai): Boolean;
- End;
- function MatchInstruction(const instr: tai; const op: TCommonAsmOps; const cond: TAsmConds; const postfix: TOpPostfixes): boolean;
- function MatchInstruction(const instr: tai; const op: TAsmOp; const cond: TAsmConds; const postfix: TOpPostfixes): boolean;
- {$ifdef AARCH64}
- function MatchInstruction(const instr: tai; const ops : array of TAsmOp; const postfix: TOpPostfixes): boolean;
- {$endif AARCH64}
- function MatchInstruction(const instr: tai; const op: TAsmOp; const postfix: TOpPostfixes): boolean;
- function RefsEqual(const r1, r2: treference): boolean;
- function MatchOperand(const oper: TOper; const reg: TRegister): boolean; inline;
- function MatchOperand(const oper1: TOper; const oper2: TOper): boolean; inline;
- function MatchOperand(const oper: TOper; const a: TCGInt): boolean; inline;
- Implementation
- uses
- cutils,verbose,globals,
- systems,
- cpuinfo,
- cgobj,procinfo,
- aasmbase,aasmdata,itcpugas;
- {$ifdef DEBUG_AOPTCPU}
- const
- SPeepholeOptimization: shortstring = 'Peephole Optimization: ';
- procedure TARMAsmOptimizer.DebugMsg(const s: string;p : tai);
- begin
- asml.insertbefore(tai_comment.Create(strpnew(s)), p);
- end;
- {$else DEBUG_AOPTCPU}
- { Empty strings help the optimizer to remove string concatenations that won't
- ever appear to the user on release builds. [Kit] }
- const
- SPeepholeOptimization = '';
- procedure TARMAsmOptimizer.DebugMsg(const s: string;p : tai);inline;
- begin
- end;
- {$endif DEBUG_AOPTCPU}
- function MatchInstruction(const instr: tai; const op: TCommonAsmOps; const cond: TAsmConds; const postfix: TOpPostfixes): boolean;
- begin
- result :=
- (instr.typ = ait_instruction) and
- ((op = []) or ((taicpu(instr).opcode<=LastCommonAsmOp) and (taicpu(instr).opcode in op))) and
- ((cond = []) or (taicpu(instr).condition in cond)) and
- ((postfix = []) or (taicpu(instr).oppostfix in postfix));
- end;
- function MatchInstruction(const instr: tai; const op: TAsmOp; const cond: TAsmConds; const postfix: TOpPostfixes): boolean;
- begin
- result :=
- (instr.typ = ait_instruction) and
- (taicpu(instr).opcode = op) and
- ((cond = []) or (taicpu(instr).condition in cond)) and
- ((postfix = []) or (taicpu(instr).oppostfix in postfix));
- end;
- {$ifdef AARCH64}
- function MatchInstruction(const instr: tai; const ops : array of TAsmOp; const postfix: TOpPostfixes): boolean;
- var
- op : TAsmOp;
- begin
- result:=false;
- if instr.typ <> ait_instruction then
- exit;
- for op in ops do
- begin
- if (taicpu(instr).opcode = op) and
- ((postfix = []) or (taicpu(instr).oppostfix in postfix)) then
- begin
- result:=true;
- exit;
- end;
- end;
- end;
- {$endif AARCH64}
- function MatchInstruction(const instr: tai; const op: TAsmOp; const postfix: TOpPostfixes): boolean;
- begin
- result :=
- (instr.typ = ait_instruction) and
- (taicpu(instr).opcode = op) and
- ((postfix = []) or (taicpu(instr).oppostfix in postfix));
- end;
- function MatchOperand(const oper: TOper; const reg: TRegister): boolean; inline;
- begin
- result := (oper.typ = top_reg) and (oper.reg = reg);
- end;
- function RefsEqual(const r1, r2: treference): boolean;
- begin
- refsequal :=
- (r1.offset = r2.offset) and
- (r1.base = r2.base) and
- (r1.index = r2.index) and (r1.scalefactor = r2.scalefactor) and
- (r1.symbol=r2.symbol) and (r1.refaddr = r2.refaddr) and
- (r1.relsymbol = r2.relsymbol) and
- {$ifdef ARM}
- (r1.signindex = r2.signindex) and
- {$endif ARM}
- (r1.shiftimm = r2.shiftimm) and
- (r1.addressmode = r2.addressmode) and
- (r1.shiftmode = r2.shiftmode) and
- (r1.volatility=[]) and
- (r2.volatility=[]);
- end;
- function MatchOperand(const oper1: TOper; const oper2: TOper): boolean; inline;
- begin
- result := oper1.typ = oper2.typ;
- if result then
- case oper1.typ of
- top_const:
- Result:=oper1.val = oper2.val;
- top_reg:
- Result:=oper1.reg = oper2.reg;
- top_conditioncode:
- Result:=oper1.cc = oper2.cc;
- top_realconst:
- Result:=oper1.val_real = oper2.val_real;
- top_ref:
- Result:=RefsEqual(oper1.ref^, oper2.ref^);
- else Result:=false;
- end
- end;
- function MatchOperand(const oper: TOper; const a: TCGInt): boolean; inline;
- begin
- result := (oper.typ = top_const) and (oper.val = a);
- end;
- {$ifdef AARCH64}
- function TARMAsmOptimizer.USxtOp2Op(var p,hp1: tai; shiftmode: tshiftmode): Boolean;
- var
- so: tshifterop;
- opoffset: Integer;
- begin
- Result:=false;
- if (taicpu(p).ops=2) and
- ((MatchInstruction(hp1, [A_ADD,A_SUB], [C_None], [PF_None,PF_S]) and
- (taicpu(hp1).ops=3) and
- MatchOperand(taicpu(hp1).oper[2]^, taicpu(p).oper[0]^.reg) and
- not(MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg))) or
- (MatchInstruction(hp1, [A_CMP,A_CMN], [C_None], [PF_None]) and
- (taicpu(hp1).ops=2) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg))
- ) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole '+gas_op2str[taicpu(p).opcode]+gas_op2str[taicpu(hp1).opcode]+'2'+gas_op2str[taicpu(hp1).opcode]+' done', p);
- AllocRegBetween(taicpu(p).oper[1]^.reg,p,hp1,UsedRegs);
- if MatchInstruction(hp1, [A_CMP,A_CMN], [C_None], [PF_None]) then
- opoffset:=0
- else
- opoffset:=1;
- taicpu(hp1).loadReg(opoffset+1,taicpu(p).oper[1]^.reg);
- if not(shiftmode in [SM_SXTX,SM_UXTX,SM_LSL]) then
- setsubreg(taicpu(hp1).oper[opoffset+1]^.reg,R_SUBD);
- taicpu(hp1).ops:=opoffset+3;
- shifterop_reset(so);
- so.shiftmode:=shiftmode;
- so.shiftimm:=0;
- taicpu(hp1).loadshifterop(opoffset+2,so);
- result:=RemoveCurrentP(p);
- end;
- end;
- {$endif AARCH64}
- function TARMAsmOptimizer.GetNextInstructionUsingReg(Current: tai;
- Out Next: tai; const reg: TRegister): Boolean;
- var
- gniResult: Boolean;
- begin
- Next:=Current;
- Result := False;
- repeat
- gniResult:=GetNextInstruction(Next,Next);
- if gniResult and RegInInstruction(reg,Next) then
- { Found something }
- Exit(True);
- until not gniResult or
- not(cs_opt_level3 in current_settings.optimizerswitches) or
- (Next.typ<>ait_instruction) or
- is_calljmp(taicpu(Next).opcode)
- {$ifdef ARM}
- or RegModifiedByInstruction(NR_PC,Next)
- {$endif ARM}
- ;
- end;
- function TARMAsmOptimizer.RemoveSuperfluousMove(const p: tai; movp: tai; const optimizer: string):boolean;
- var
- alloc,
- dealloc : tai_regalloc;
- hp1 : tai;
- begin
- Result:=false;
- if MatchInstruction(movp, A_MOV, [taicpu(p).condition], [PF_None]) and
- { We can't optimize if there is a shiftop }
- (taicpu(movp).ops=2) and
- MatchOperand(taicpu(movp).oper[1]^, taicpu(p).oper[0]^.reg) and
- { don't mess with moves to fp }
- (taicpu(movp).oper[0]^.reg<>current_procinfo.framepointer) and
- { the destination register of the mov might not be used beween p and movp }
- not(RegUsedBetween(taicpu(movp).oper[0]^.reg,p,movp)) and
- {$ifdef ARM}
- { PC should be changed only by moves }
- (taicpu(movp).oper[0]^.reg<>NR_PC) and
- { cb[n]z are thumb instructions which require specific registers, with no wide forms }
- (taicpu(p).opcode<>A_CBZ) and
- (taicpu(p).opcode<>A_CBNZ) and
- { There is a special requirement for MUL and MLA, oper[0] and oper[1] are not allowed to be the same }
- not (
- (taicpu(p).opcode in [A_MLA, A_MUL]) and
- (taicpu(p).oper[1]^.reg = taicpu(movp).oper[0]^.reg) and
- (current_settings.cputype < cpu_armv6)
- ) and
- {$endif ARM}
- { Take care to only do this for instructions which REALLY load to the first register.
- Otherwise
- str reg0, [reg1]
- mov reg2, reg0
- will be optimized to
- str reg2, [reg1]
- }
- RegLoadedWithNewValue(taicpu(p).oper[0]^.reg, p) then
- begin
- dealloc:=FindRegDeAlloc(taicpu(p).oper[0]^.reg,tai(movp.Next));
- if assigned(dealloc) then
- begin
- DebugMsg('Peephole '+optimizer+' removed superfluous mov', movp);
- result:=true;
- { taicpu(p).oper[0]^.reg is not used anymore, try to find its allocation
- and remove it if possible }
- asml.Remove(dealloc);
- alloc:=FindRegAllocBackward(taicpu(p).oper[0]^.reg,tai(p.previous));
- if assigned(alloc) then
- begin
- asml.Remove(alloc);
- alloc.free;
- dealloc.free;
- end
- else
- asml.InsertAfter(dealloc,p);
- AllocRegBetween(taicpu(movp).oper[0]^.reg,p,movp,UsedRegs);
- { finally get rid of the mov }
- taicpu(p).loadreg(0,taicpu(movp).oper[0]^.reg);
- { Remove preindexing and postindexing for LDR in some cases.
- For example:
- ldr reg2,[reg1, xxx]!
- mov reg1,reg2
- must be translated to:
- ldr reg1,[reg1, xxx]
- Preindexing must be removed there, since the same register is used as the base and as the target.
- Such case is not allowed for ARM CPU and produces crash. }
- if (taicpu(p).opcode = A_LDR) and (taicpu(p).oper[1]^.typ = top_ref)
- and (taicpu(movp).oper[0]^.reg = taicpu(p).oper[1]^.ref^.base)
- then
- taicpu(p).oper[1]^.ref^.addressmode:=AM_OFFSET;
- asml.remove(movp);
- movp.free;
- end;
- end;
- end;
- function TARMAsmOptimizer.RedundantMovProcess(var p: tai; var hp1: tai):boolean;
- var
- I: Integer;
- current_hp, next_hp: tai;
- LDRChange: Boolean;
- begin
- Result:=false;
- {
- change
- mov r1, r0
- add r1, r1, #1
- to
- add r1, r0, #1
- Todo: Make it work for mov+cmp too
- CAUTION! If this one is successful p might not be a mov instruction anymore!
- }
- if (taicpu(p).ops = 2) and
- (taicpu(p).oper[1]^.typ = top_reg) and
- (taicpu(p).oppostfix = PF_NONE) then
- begin
- if
- MatchInstruction(hp1, [A_ADD, A_ADC,
- {$ifdef ARM}
- A_RSB, A_RSC,
- {$endif ARM}
- A_SUB, A_SBC,
- A_AND, A_BIC, A_EOR, A_ORR, A_MOV, A_MVN],
- [taicpu(p).condition], []) and
- { MOV and MVN might only have 2 ops }
- (taicpu(hp1).ops >= 2) and
- MatchOperand(taicpu(p).oper[0]^, taicpu(hp1).oper[0]^.reg) and
- (taicpu(hp1).oper[1]^.typ = top_reg) and
- (
- (taicpu(hp1).ops = 2) or
- (taicpu(hp1).oper[2]^.typ in [top_reg, top_const, top_shifterop])
- ) and
- {$ifdef AARCH64}
- (taicpu(p).oper[1]^.reg<>NR_SP) and
- { in this case you have to transform it to movk or the like }
- (getsupreg(taicpu(p).oper[1]^.reg)<>RS_XZR) and
- {$endif AARCH64}
- not(RegUsedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- { When we get here we still don't know if the registers match }
- for I:=1 to 2 do
- {
- If the first loop was successful p will be replaced with hp1.
- The checks will still be ok, because all required information
- will also be in hp1 then.
- }
- if (taicpu(hp1).ops > I) and
- MatchOperand(taicpu(p).oper[0]^, taicpu(hp1).oper[I]^.reg)
- {$ifdef ARM}
- { prevent certain combinations on thumb(2), this is only a safe approximation }
- and (not(GenerateThumbCode or GenerateThumb2Code) or
- ((getsupreg(taicpu(p).oper[1]^.reg)<>RS_R13) and
- (getsupreg(taicpu(p).oper[1]^.reg)<>RS_R15)))
- {$endif ARM}
- then
- begin
- DebugMsg('Peephole RedundantMovProcess done', hp1);
- taicpu(hp1).oper[I]^.reg := taicpu(p).oper[1]^.reg;
- if p<>hp1 then
- begin
- asml.remove(p);
- p.free;
- p:=hp1;
- Result:=true;
- end;
- end;
- if Result then Exit;
- end
- { Change: Change:
- mov r1, r0 mov r1, r0
- ... ...
- ldr/str r2, [r1, etc.] mov r2, r1
- To: To:
- ldr/str r2, [r0, etc.] mov r2, r0
- }
- else if (taicpu(p).condition = C_None) and (taicpu(p).oper[1]^.typ = top_reg)
- {$ifdef ARM}
- and not (getsupreg(taicpu(p).oper[0]^.reg) in [RS_PC, RS_R14, RS_STACK_POINTER_REG])
- and (getsupreg(taicpu(p).oper[1]^.reg) <> RS_PC)
- { Thumb does not support references with base and index one being SP }
- and (not(GenerateThumbCode) or (getsupreg(taicpu(p).oper[1]^.reg) <> RS_STACK_POINTER_REG))
- {$endif ARM}
- {$ifdef AARCH64}
- and (getsupreg(taicpu(p).oper[0]^.reg) <> RS_STACK_POINTER_REG)
- {$endif AARCH64}
- then
- begin
- current_hp := p;
- TransferUsedRegs(TmpUsedRegs);
- { Search local instruction block }
- while GetNextInstruction(current_hp, next_hp) and (next_hp <> BlockEnd) and (next_hp.typ = ait_instruction) do
- begin
- UpdateUsedRegs(TmpUsedRegs, tai(current_hp.Next));
- LDRChange := False;
- if (taicpu(next_hp).opcode in [A_LDR,A_STR]) and (taicpu(next_hp).ops = 2)
- {$ifdef AARCH64}
- { If r0 is the zero register, then this sequence of instructions will cause
- an access violation, but that's better than an assembler error caused by
- changing r0 to xzr inside the reference (Where it's illegal). [Kit] }
- and (getsupreg(taicpu(p).oper[1]^.reg) <> RS_XZR)
- {$endif AARCH64}
- then
- begin
- { Change the registers from r1 to r0 }
- if (taicpu(next_hp).oper[1]^.ref^.base = taicpu(p).oper[0]^.reg) and
- {$ifdef ARM}
- { This optimisation conflicts with something and raises
- an access violation - needs further investigation. [Kit] }
- (taicpu(next_hp).opcode <> A_LDR) and
- {$endif ARM}
- { Don't mess around with the base register if the
- reference is pre- or post-indexed }
- (taicpu(next_hp).oper[1]^.ref^.addressmode = AM_OFFSET) then
- begin
- taicpu(next_hp).oper[1]^.ref^.base := taicpu(p).oper[1]^.reg;
- LDRChange := True;
- end;
- if taicpu(next_hp).oper[1]^.ref^.index = taicpu(p).oper[0]^.reg then
- begin
- taicpu(next_hp).oper[1]^.ref^.index := taicpu(p).oper[1]^.reg;
- LDRChange := True;
- end;
- if LDRChange then
- DebugMsg('Peephole Optimization: ' + std_regname(taicpu(p).oper[0]^.reg) + ' = ' + std_regname(taicpu(p).oper[1]^.reg) + ' (MovLdr2Ldr 1)', next_hp);
- { Drop out if we're dealing with pre-indexed references }
- if (taicpu(next_hp).oper[1]^.ref^.addressmode = AM_PREINDEXED) and
- (
- RegInRef(taicpu(p).oper[0]^.reg, taicpu(next_hp).oper[1]^.ref^) or
- RegInRef(taicpu(p).oper[1]^.reg, taicpu(next_hp).oper[1]^.ref^)
- ) then
- begin
- { Remember to update register allocations }
- if LDRChange then
- AllocRegBetween(taicpu(p).oper[1]^.reg, p, next_hp, UsedRegs);
- Break;
- end;
- { The register being stored can be potentially changed (as long as it's not the stack pointer) }
- if (taicpu(next_hp).opcode = A_STR) and (getsupreg(taicpu(p).oper[1]^.reg) <> RS_STACK_POINTER_REG) and
- MatchOperand(taicpu(next_hp).oper[0]^, taicpu(p).oper[0]^.reg) then
- begin
- DebugMsg('Peephole Optimization: ' + std_regname(taicpu(p).oper[0]^.reg) + ' = ' + std_regname(taicpu(p).oper[1]^.reg) + ' (MovLdr2Ldr 2)', next_hp);
- taicpu(next_hp).oper[0]^.reg := taicpu(p).oper[1]^.reg;
- LDRChange := True;
- end;
- if LDRChange and (getsupreg(taicpu(p).oper[1]^.reg) <> RS_STACK_POINTER_REG) then
- begin
- AllocRegBetween(taicpu(p).oper[1]^.reg, p, next_hp, UsedRegs);
- if (taicpu(p).oppostfix = PF_None) and
- (
- (
- (taicpu(next_hp).opcode = A_LDR) and
- MatchOperand(taicpu(next_hp).oper[0]^, taicpu(p).oper[0]^.reg)
- ) or
- not RegUsedAfterInstruction(taicpu(p).oper[0]^.reg, next_hp, TmpUsedRegs)
- ) and
- { Double-check to see if the old registers were actually
- changed (e.g. if the super registers matched, but not
- the sizes, they won't be changed). }
- (
- (taicpu(next_hp).opcode = A_LDR) or
- not RegInOp(taicpu(p).oper[0]^.reg, taicpu(next_hp).oper[0]^)
- ) and
- not RegInRef(taicpu(p).oper[0]^.reg, taicpu(next_hp).oper[1]^.ref^) then
- begin
- DebugMsg('Peephole Optimization: RedundantMovProcess 2a done', p);
- RemoveCurrentP(p);
- Result := True;
- Exit;
- end;
- end;
- end
- else if (taicpu(next_hp).opcode = A_MOV) and (taicpu(next_hp).oppostfix = PF_None) and
- (taicpu(next_hp).ops = 2) then
- begin
- if MatchOperand(taicpu(next_hp).oper[0]^, taicpu(p).oper[0]^.reg) then
- begin
- { mov r0,r1; mov r1,r1 - remove second MOV here so
- so "RedundantMovProcess 2b" doesn't get erroneously
- applied }
- if MatchOperand(taicpu(next_hp).oper[0]^, taicpu(next_hp).oper[1]^.reg) then
- begin
- DebugMsg(SPeepholeOptimization + 'Mov2None 2a done', next_hp);
- if (next_hp = hp1) then
- { Don't let hp1 become a dangling pointer }
- hp1 := nil;
- asml.Remove(next_hp);
- next_hp.Free;
- Continue;
- end;
- { Found another mov that writes entirely to the register }
- if RegUsedBetween(taicpu(p).oper[0]^.reg, p, next_hp) then
- begin
- { Register was used beforehand }
- if MatchOperand(taicpu(next_hp).oper[1]^, taicpu(p).oper[1]^.reg) then
- begin
- { This MOV is exactly the same as the first one.
- Since none of the registers have changed value
- at this point, we can remove it. }
- DebugMsg(SPeepholeOptimization + 'RedundantMovProcess 3a done', next_hp);
- if (next_hp = hp1) then
- { Don't let hp1 become a dangling pointer }
- hp1 := nil;
- asml.Remove(next_hp);
- next_hp.Free;
- { We still have the original p, so we can continue optimising;
- if it was -O2 or below, this instruction appeared immediately
- after the first MOV, so we're technically not looking more
- than one instruction ahead after it's removed! [Kit] }
- Continue;
- end
- else
- { Register changes value - drop out }
- Break;
- end;
- { We can delete the first MOV (only if the second MOV is unconditional) }
- {$ifdef ARM}
- if (taicpu(p).oppostfix = PF_None) and
- (taicpu(next_hp).condition = C_None) then
- {$endif ARM}
- begin
- DebugMsg('Peephole Optimization: RedundantMovProcess 2b done', p);
- RemoveCurrentP(p);
- Result := True;
- end;
- Exit;
- end
- else if MatchOperand(taicpu(next_hp).oper[1]^, taicpu(p).oper[0]^.reg) then
- begin
- if MatchOperand(taicpu(next_hp).oper[0]^, taicpu(p).oper[1]^.reg)
- { Be careful - if the entire register is not used, removing this
- instruction will leave the unused part uninitialised }
- {$ifdef AARCH64}
- and (getsubreg(taicpu(p).oper[1]^.reg) = R_SUBQ)
- {$endif AARCH64}
- then
- begin
- { Instruction will become mov r1,r1 }
- DebugMsg(SPeepholeOptimization + 'Mov2None 2 done', next_hp);
- { Allocate r1 between the instructions; not doing
- so may cause problems when removing superfluous
- MOVs later (i38055) }
- AllocRegBetween(taicpu(p).oper[1]^.reg, p, next_hp, UsedRegs);
- if (next_hp = hp1) then
- { Don't let hp1 become a dangling pointer }
- hp1 := nil;
- asml.Remove(next_hp);
- next_hp.Free;
- Continue;
- end;
- { Change the old register (checking the first operand again
- forces it to be left alone if the full register is not
- used, lest mov w1,w1 gets optimised out by mistake. [Kit] }
- {$ifdef AARCH64}
- if not MatchOperand(taicpu(next_hp).oper[0]^, taicpu(p).oper[1]^.reg) then
- {$endif AARCH64}
- begin
- DebugMsg(SPeepholeOptimization + std_regname(taicpu(p).oper[0]^.reg) + ' = ' + std_regname(taicpu(p).oper[1]^.reg) + ' (MovMov2Mov 2)', next_hp);
- taicpu(next_hp).oper[1]^.reg := taicpu(p).oper[1]^.reg;
- AllocRegBetween(taicpu(p).oper[1]^.reg, p, next_hp, UsedRegs);
- { If this was the only reference to the old register,
- then we can remove the original MOV now }
- if (taicpu(p).oppostfix = PF_None) and
- { A bit of a hack - sometimes registers aren't tracked properly, so do not
- remove if the register was apparently not allocated when its value is
- first set at the MOV command (this is especially true for the stack
- register). [Kit] }
- (getsupreg(taicpu(p).oper[1]^.reg) <> RS_STACK_POINTER_REG) and
- RegInUsedRegs(taicpu(p).oper[0]^.reg, UsedRegs) and
- not RegUsedAfterInstruction(taicpu(p).oper[0]^.reg, next_hp, TmpUsedRegs) then
- begin
- DebugMsg(SPeepholeOptimization + 'RedundantMovProcess 2c done', p);
- RemoveCurrentP(p);
- Result := True;
- Exit;
- end;
- end;
- end;
- end;
- { On low optimisation settions, don't search more than one instruction ahead }
- if not(cs_opt_level3 in current_settings.optimizerswitches) or
- { Stop at procedure calls and jumps }
- is_calljmp(taicpu(next_hp).opcode) or
- { If the read register has changed value, or the MOV
- destination register has been used, drop out }
- RegInInstruction(taicpu(p).oper[0]^.reg, next_hp) or
- RegModifiedByInstruction(taicpu(p).oper[1]^.reg, next_hp) then
- Break;
- current_hp := next_hp;
- end;
- end;
- end;
- end;
- function TARMAsmOptimizer.OptPass1UXTB(var p : tai) : Boolean;
- var
- hp1, hp2: tai;
- so: tshifterop;
- begin
- Result:=false;
- if GetNextInstructionUsingReg(p,hp1,taicpu(p).oper[0]^.reg) then
- begin
- {
- change
- uxtb reg2,reg1
- strb reg2,[...]
- dealloc reg2
- to
- strb reg1,[...]
- }
- if MatchInstruction(p, taicpu(p).opcode, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_STR, [C_None], [PF_B]) and
- assigned(FindRegDealloc(taicpu(p).oper[0]^.reg,tai(hp1.Next))) and
- { the reference in strb might not use reg2 }
- not(RegInRef(taicpu(p).oper[0]^.reg,taicpu(hp1).oper[1]^.ref^)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole UxtbStrb2Strb done', p);
- taicpu(hp1).loadReg(0,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {
- change
- uxtb reg2,reg1
- uxth reg3,reg2
- dealloc reg2
- to
- uxtb reg3,reg1
- }
- else if MatchInstruction(p, A_UXTB, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_UXTH, [C_None], [PF_None]) and
- (taicpu(hp1).ops = 2) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole UxtbUxth2Uxtb done', p);
- AllocRegBetween(taicpu(hp1).oper[0]^.reg,p,hp1,UsedRegs);
- taicpu(p).loadReg(0,taicpu(hp1).oper[0]^.reg);
- asml.remove(hp1);
- hp1.free;
- result:=true;
- end
- {
- change
- uxtb reg2,reg1
- uxtb reg3,reg2
- dealloc reg2
- to
- uxtb reg3,reg1
- }
- else if MatchInstruction(p, A_UXTB, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_UXTB, [C_None], [PF_None]) and
- (taicpu(hp1).ops = 2) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole UxtbUxtb2Uxtb done', p);
- AllocRegBetween(taicpu(hp1).oper[0]^.reg,p,hp1,UsedRegs);
- taicpu(p).loadReg(0,taicpu(hp1).oper[0]^.reg);
- asml.remove(hp1);
- hp1.free;
- result:=true;
- end
- {
- change
- uxtb reg2,reg1
- and reg3,reg2,#0x*FF
- dealloc reg2
- to
- uxtb reg3,reg1
- }
- else if MatchInstruction(p, A_UXTB, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_AND, [C_None], [PF_None]) and
- (taicpu(hp1).ops=3) and
- (taicpu(hp1).oper[2]^.typ=top_const) and
- ((taicpu(hp1).oper[2]^.val and $FF)=$FF) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole UxtbAndImm2Uxtb done', p);
- taicpu(hp1).opcode:=A_UXTB;
- taicpu(hp1).ops:=2;
- taicpu(hp1).loadReg(1,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {$ifdef AARCH64}
- else if USxtOp2Op(p,hp1,SM_UXTB) then
- Result:=true
- {$endif AARCH64}
- else if RemoveSuperfluousMove(p, hp1, 'UxtbMov2Uxtb') then
- Result:=true;
- end;
- end;
- function TARMAsmOptimizer.OptPass1UXTH(var p : tai) : Boolean;
- var
- hp1: tai;
- so: tshifterop;
- begin
- Result:=false;
- if GetNextInstructionUsingReg(p,hp1,taicpu(p).oper[0]^.reg) then
- begin
- {
- change
- uxth reg2,reg1
- strh reg2,[...]
- dealloc reg2
- to
- strh reg1,[...]
- }
- if MatchInstruction(p, taicpu(p).opcode, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_STR, [C_None], [PF_H]) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { the reference in strb might not use reg2 }
- not(RegInRef(taicpu(p).oper[0]^.reg,taicpu(hp1).oper[1]^.ref^)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole UXTHStrh2Strh done', p);
- taicpu(hp1).loadReg(0,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {
- change
- uxth reg2,reg1
- uxth reg3,reg2
- dealloc reg2
- to
- uxth reg3,reg1
- }
- else if MatchInstruction(p, A_UXTH, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_UXTH, [C_None], [PF_None]) and
- (taicpu(hp1).ops=2) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole UxthUxth2Uxth done', p);
- AllocRegBetween(taicpu(p).oper[1]^.reg,p,hp1,UsedRegs);
- taicpu(hp1).opcode:=A_UXTH;
- taicpu(hp1).loadReg(1,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {
- change
- uxth reg2,reg1
- and reg3,reg2,#65535
- dealloc reg2
- to
- uxth reg3,reg1
- }
- else if MatchInstruction(p, A_UXTH, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_AND, [C_None], [PF_None]) and
- (taicpu(hp1).ops=3) and
- (taicpu(hp1).oper[2]^.typ=top_const) and
- ((taicpu(hp1).oper[2]^.val and $FFFF)=$FFFF) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole UxthAndImm2Uxth done', p);
- taicpu(hp1).opcode:=A_UXTH;
- taicpu(hp1).ops:=2;
- taicpu(hp1).loadReg(1,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {$ifdef AARCH64}
- else if USxtOp2Op(p,hp1,SM_UXTH) then
- Result:=true
- {$endif AARCH64}
- else if RemoveSuperfluousMove(p, hp1, 'UxthMov2Data') then
- Result:=true;
- end;
- end;
- function TARMAsmOptimizer.OptPass1SXTB(var p : tai) : Boolean;
- var
- hp1, hp2: tai;
- so: tshifterop;
- begin
- Result:=false;
- if GetNextInstructionUsingReg(p,hp1,taicpu(p).oper[0]^.reg) then
- begin
- {
- change
- sxtb reg2,reg1
- strb reg2,[...]
- dealloc reg2
- to
- strb reg1,[...]
- }
- if MatchInstruction(p, taicpu(p).opcode, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_STR, [C_None], [PF_B]) and
- assigned(FindRegDealloc(taicpu(p).oper[0]^.reg,tai(hp1.Next))) and
- { the reference in strb might not use reg2 }
- not(RegInRef(taicpu(p).oper[0]^.reg,taicpu(hp1).oper[1]^.ref^)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole SxtbStrb2Strb done', p);
- taicpu(hp1).loadReg(0,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {
- change
- sxtb reg2,reg1
- sxth reg3,reg2
- dealloc reg2
- to
- sxtb reg3,reg1
- }
- else if MatchInstruction(p, A_SXTB, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_SXTH, [C_None], [PF_None]) and
- (taicpu(hp1).ops = 2) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole SxtbSxth2Sxtb done', p);
- AllocRegBetween(taicpu(hp1).oper[0]^.reg,p,hp1,UsedRegs);
- taicpu(p).loadReg(0,taicpu(hp1).oper[0]^.reg);
- asml.remove(hp1);
- hp1.free;
- result:=true;
- end
- {
- change
- sxtb reg2,reg1
- sxtb reg3,reg2
- dealloc reg2
- to
- uxtb reg3,reg1
- }
- else if MatchInstruction(p, A_SXTB, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_SXTB, [C_None], [PF_None]) and
- (taicpu(hp1).ops = 2) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole SxtbSxtb2Sxtb done', p);
- AllocRegBetween(taicpu(hp1).oper[0]^.reg,p,hp1,UsedRegs);
- taicpu(p).loadReg(0,taicpu(hp1).oper[0]^.reg);
- asml.remove(hp1);
- hp1.free;
- result:=true;
- end
- {
- change
- sxtb reg2,reg1
- and reg3,reg2,#0x*FF
- dealloc reg2
- to
- uxtb reg3,reg1
- }
- else if MatchInstruction(p, A_SXTB, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_AND, [C_None], [PF_None]) and
- (taicpu(hp1).ops=3) and
- (taicpu(hp1).oper[2]^.typ=top_const) and
- ((taicpu(hp1).oper[2]^.val and $FF)=$FF) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole SxtbAndImm2Uxtb done', p);
- taicpu(hp1).opcode:=A_UXTB;
- taicpu(hp1).ops:=2;
- taicpu(hp1).loadReg(1,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {$ifdef AARCH64}
- else if USxtOp2Op(p,hp1,SM_SXTB) then
- Result:=true
- {$endif AARCH64}
- else if GetNextInstructionUsingReg(p, hp1, taicpu(p).oper[0]^.reg) and
- RemoveSuperfluousMove(p, hp1, 'SxtbMov2Sxtb') then
- Result:=true;
- end;
- end;
- function TARMAsmOptimizer.OptPass1SXTH(var p : tai) : Boolean;
- var
- hp1: tai;
- so: tshifterop;
- begin
- Result:=false;
- if GetNextInstructionUsingReg(p,hp1,taicpu(p).oper[0]^.reg) then
- begin
- {
- change
- sxth reg2,reg1
- strh reg2,[...]
- dealloc reg2
- to
- strh reg1,[...]
- }
- if MatchInstruction(p, taicpu(p).opcode, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_STR, [C_None], [PF_H]) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { the reference in strb might not use reg2 }
- not(RegInRef(taicpu(p).oper[0]^.reg,taicpu(hp1).oper[1]^.ref^)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole SxthStrh2Strh done', p);
- taicpu(hp1).loadReg(0,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {
- change
- sxth reg2,reg1
- sxth reg3,reg2
- dealloc reg2
- to
- sxth reg3,reg1
- }
- else if MatchInstruction(p, A_SXTH, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_SXTH, [C_None], [PF_None]) and
- (taicpu(hp1).ops=2) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole SxthSxth2Sxth done', p);
- AllocRegBetween(taicpu(p).oper[1]^.reg,p,hp1,UsedRegs);
- taicpu(hp1).opcode:=A_SXTH;
- taicpu(hp1).loadReg(1,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {$ifdef AARCH64}
- {
- change
- sxth reg2,reg1
- sxtw reg3,reg2
- dealloc reg2
- to
- sxth reg3,reg1
- }
- else if MatchInstruction(p, A_SXTH, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_SXTW, [C_None], [PF_None]) and
- (taicpu(hp1).ops=2) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole SxthSxtw2Sxth done', p);
- AllocRegBetween(taicpu(p).oper[1]^.reg,p,hp1,UsedRegs);
- taicpu(hp1).opcode:=A_SXTH;
- taicpu(hp1).loadReg(1,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {$endif AARCH64}
- {
- change
- sxth reg2,reg1
- and reg3,reg2,#65535
- dealloc reg2
- to
- uxth reg3,reg1
- }
- else if MatchInstruction(p, A_SXTH, [C_None], [PF_None]) and
- (taicpu(p).ops=2) and
- MatchInstruction(hp1, A_AND, [C_None], [PF_None]) and
- (taicpu(hp1).ops=3) and
- (taicpu(hp1).oper[2]^.typ=top_const) and
- ((taicpu(hp1).oper[2]^.val and $FFFF)=$FFFF) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole SxthAndImm2Uxth done', p);
- taicpu(hp1).opcode:=A_UXTH;
- taicpu(hp1).ops:=2;
- taicpu(hp1).loadReg(1,taicpu(p).oper[1]^.reg);
- result:=RemoveCurrentP(p);
- end
- {$ifdef AARCH64}
- else if USxtOp2Op(p,hp1,SM_SXTH) then
- Result:=true
- {$endif AARCH64}
- else if GetNextInstructionUsingReg(p, hp1, taicpu(p).oper[0]^.reg) and
- RemoveSuperfluousMove(p, hp1, 'SxthMov2Sxth') then
- Result:=true;
- end;
- end;
- function TARMAsmOptimizer.OptPreSBFXUBFX(var p: tai): Boolean;
- begin
- Result := False;
- { Convert:
- s/ubfx reg1,reg2,#0,#64 (or #32 for 32-bit registers)
- To:
- mov reg1,reg2
- }
- if (taicpu(p).oper[2]^.val = 0) and
- {$ifdef AARCH64}
- (
- (
- (getsubreg(taicpu(p).oper[0]^.reg) = R_SUBQ) and
- (taicpu(p).oper[3]^.val = 64)
- ) or
- (
- (getsubreg(taicpu(p).oper[0]^.reg) = R_SUBD) and
- (taicpu(p).oper[3]^.val = 32)
- )
- )
- {$else AARCH64}
- (taicpu(p).oper[3]^.val = 32)
- {$endif AARCH64}
- then
- begin
- DebugMsg(SPeepholeOptimization + 'SBFX or UBFX -> MOV (full bitfield extract)', p);
- taicpu(p).opcode := A_MOV;
- taicpu(p).ops := 2;
- taicpu(p).clearop(2);
- taicpu(p).clearop(3);
- Result := True;
- Exit;
- end;
- end;
- function TARMAsmOptimizer.OptPass1LDR(var p : tai) : Boolean;
- var
- hp1: tai;
- Reference: TReference;
- NewOp: TAsmOp;
- begin
- Result := False;
- if (taicpu(p).ops <> 2) or (taicpu(p).condition <> C_None) then
- Exit;
- Reference := taicpu(p).oper[1]^.ref^;
- if (Reference.addressmode = AM_OFFSET) and
- not RegInRef(taicpu(p).oper[0]^.reg, Reference) and
- { Delay calling GetNextInstruction for as long as possible }
- GetNextInstruction(p, hp1) and
- (hp1.typ = ait_instruction) and
- (taicpu(hp1).condition = C_None) and
- (taicpu(hp1).oppostfix = taicpu(p).oppostfix) then
- begin
- if (taicpu(hp1).opcode = A_STR) and
- RefsEqual(taicpu(hp1).oper[1]^.ref^, Reference) and
- (getregtype(taicpu(p).oper[0]^.reg) = getregtype(taicpu(hp1).oper[0]^.reg)) then
- begin
- { With:
- ldr reg1,[ref]
- str reg2,[ref]
- If reg1 = reg2, Remove str
- }
- if taicpu(p).oper[0]^.reg = taicpu(hp1).oper[0]^.reg then
- begin
- DebugMsg(SPeepholeOptimization + 'Removed redundant store instruction (load/store -> load/nop)', hp1);
- RemoveInstruction(hp1);
- Result := True;
- Exit;
- end;
- end
- else if (taicpu(hp1).opcode = A_LDR) and
- RefsEqual(taicpu(hp1).oper[1]^.ref^, Reference) then
- begin
- { With:
- ldr reg1,[ref]
- ldr reg2,[ref]
- If reg1 = reg2, delete the second ldr
- If reg1 <> reg2, changing the 2nd ldr to a mov might introduce
- a dependency, but it will likely open up new optimisations, so
- do it for now and handle any new dependencies later.
- }
- if taicpu(p).oper[0]^.reg = taicpu(hp1).oper[0]^.reg then
- begin
- DebugMsg(SPeepholeOptimization + 'Removed duplicate load instruction (load/load -> load/nop)', hp1);
- RemoveInstruction(hp1);
- Result := True;
- Exit;
- end
- else if
- (getregtype(taicpu(p).oper[0]^.reg) = R_INTREGISTER) and
- (getregtype(taicpu(hp1).oper[0]^.reg) = R_INTREGISTER) and
- (getsubreg(taicpu(p).oper[0]^.reg) = getsubreg(taicpu(hp1).oper[0]^.reg)) then
- begin
- DebugMsg(SPeepholeOptimization + 'Changed second ldr' + oppostfix2str[taicpu(hp1).oppostfix] + ' to mov (load/load -> load/move)', hp1);
- taicpu(hp1).opcode := A_MOV;
- taicpu(hp1).oppostfix := PF_None;
- taicpu(hp1).loadreg(1, taicpu(p).oper[0]^.reg);
- AllocRegBetween(taicpu(p).oper[0]^.reg, p, hp1, UsedRegs);
- Result := True;
- Exit;
- end;
- end;
- end;
- end;
- function TARMAsmOptimizer.OptPass1STR(var p : tai) : Boolean;
- var
- hp1: tai;
- Reference: TReference;
- SizeMismatch: Boolean;
- SrcReg, DstReg: TRegister;
- NewOp: TAsmOp;
- begin
- Result := False;
- if (taicpu(p).ops <> 2) or (taicpu(p).condition <> C_None) then
- Exit;
- Reference := taicpu(p).oper[1]^.ref^;
- if (Reference.addressmode = AM_OFFSET) and
- not RegInRef(taicpu(p).oper[0]^.reg, Reference) and
- { Delay calling GetNextInstruction for as long as possible }
- GetNextInstruction(p, hp1) and
- (hp1.typ = ait_instruction) and
- (taicpu(hp1).condition = C_None) and
- (taicpu(hp1).oppostfix = taicpu(p).oppostfix) and
- (taicpu(hp1).ops>0) and (taicpu(hp1).oper[0]^.typ=top_reg) then
- begin
- { Saves constant dereferencing and makes it easier to change the size if necessary }
- SrcReg := taicpu(p).oper[0]^.reg;
- DstReg := taicpu(hp1).oper[0]^.reg;
- if (taicpu(hp1).opcode = A_LDR) and
- RefsEqual(taicpu(hp1).oper[1]^.ref^, Reference) and
- (taicpu(hp1).oper[1]^.ref^.volatility=[]) and
- (
- (taicpu(hp1).oppostfix = taicpu(p).oppostfix) or
- ((taicpu(p).oppostfix = PF_B) and (taicpu(hp1).oppostfix = PF_SB)) or
- ((taicpu(p).oppostfix = PF_H) and (taicpu(hp1).oppostfix = PF_SH))
- {$ifdef AARCH64}
- or ((taicpu(p).oppostfix = PF_W) and (taicpu(hp1).oppostfix = PF_SW))
- {$endif AARCH64}
- ) then
- begin
- { With:
- str reg1,[ref]
- ldr reg2,[ref]
- If reg1 = reg2, Remove ldr.
- If reg1 <> reg2, replace ldr with "mov reg2,reg1"
- }
- if (SrcReg = DstReg) and
- { e.g. the ldrb in strb/ldrb is not a null operation as it clears the upper 24 bits }
- (taicpu(p).oppostfix=PF_None) then
- begin
- DebugMsg(SPeepholeOptimization + 'Removed redundant load instruction (store/load -> store/nop)', hp1);
- RemoveInstruction(hp1);
- Result := True;
- Exit;
- end
- else if (getregtype(SrcReg) = R_INTREGISTER) and
- (getregtype(DstReg) = R_INTREGISTER) and
- (getsubreg(SrcReg) = getsubreg(DstReg)) then
- begin
- NewOp:=A_NONE;
- if taicpu(hp1).oppostfix=PF_None then
- NewOp:=A_MOV
- else
- {$ifdef ARM}
- if (current_settings.cputype < cpu_armv6) then
- begin
- { The zero- and sign-extension operations were only
- introduced under ARMv6 }
- case taicpu(hp1).oppostfix of
- PF_B:
- begin
- { The if-block afterwards will set the middle operand to the correct register }
- taicpu(hp1).allocate_oper(3);
- taicpu(hp1).ops := 3;
- taicpu(hp1).loadconst(2, $FF);
- NewOp := A_AND;
- end;
- PF_H:
- { ARMv5 and under doesn't have a concise way of storing the immediate $FFFF, so leave alone };
- PF_SB,
- PF_SH:
- { Do nothing - can't easily encode sign-extensions };
- else
- InternalError(2021043002);
- end;
- end
- else
- {$endif ARM}
- case taicpu(hp1).oppostfix of
- PF_B:
- NewOp := A_UXTB;
- PF_SB:
- NewOp := A_SXTB;
- PF_H:
- NewOp := A_UXTH;
- PF_SH:
- NewOp := A_SXTH;
- {$ifdef AARCH64}
- PF_SW:
- NewOp := A_SXTW;
- PF_W:
- NewOp := A_MOV;
- {$endif AARCH64}
- else
- InternalError(2021043001);
- end;
- if (NewOp<>A_None) then
- begin
- DebugMsg(SPeepholeOptimization + 'Changed ldr' + oppostfix2str[taicpu(hp1).oppostfix] + ' to ' + gas_op2str[NewOp] + ' (store/load -> store/move)', hp1);
- taicpu(hp1).oppostfix := PF_None;
- taicpu(hp1).opcode := NewOp;
- taicpu(hp1).loadreg(1, SrcReg);
- AllocRegBetween(SrcReg, p, hp1, UsedRegs);
- Result := True;
- Exit;
- end;
- end
- end
- else if (taicpu(hp1).opcode = A_STR) and
- RefsEqual(taicpu(hp1).oper[1]^.ref^, Reference) then
- begin
- { With:
- str reg1,[ref]
- str reg2,[ref]
- If reg1 <> reg2, delete the first str
- IF reg1 = reg2, delete the second str
- }
- if (SrcReg = DstReg) and (taicpu(hp1).oper[1]^.ref^.volatility=[]) then
- begin
- DebugMsg(SPeepholeOptimization + 'Removed duplicate store instruction (store/store -> store/nop)', hp1);
- RemoveInstruction(hp1);
- Result := True;
- Exit;
- end
- else if
- { Registers same byte size? }
- (tcgsize2size[reg_cgsize(SrcReg)] = tcgsize2size[reg_cgsize(DstReg)]) and
- (taicpu(p).oper[1]^.ref^.volatility=[]) then
- begin
- DebugMsg(SPeepholeOptimization + 'Removed dominated store instruction (store/store -> nop/store)', p);
- RemoveCurrentP(p, hp1);
- Result := True;
- Exit;
- end;
- end;
- end;
- end;
- function TARMAsmOptimizer.OptPass1And(var p : tai) : Boolean;
- var
- hp1, hp2: tai;
- i: longint;
- begin
- Result:=false;
- {
- optimize
- and reg2,reg1,const1
- ...
- }
- if (taicpu(p).ops>2) and
- (taicpu(p).oper[1]^.typ = top_reg) and
- (taicpu(p).oper[2]^.typ = top_const) then
- begin
- {
- change
- and reg2,reg1,const1
- ...
- and reg3,reg2,const2
- to
- and reg3,reg1,(const1 and const2)
- }
- if GetNextInstructionUsingReg(p,hp1,taicpu(p).oper[0]^.reg) and
- MatchInstruction(hp1, A_AND, [taicpu(p).condition], [PF_None]) and
- RegEndOfLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- (taicpu(hp1).oper[2]^.typ = top_const)
- {$ifdef AARCH64}
- and ((((getsubreg(taicpu(p).oper[0]^.reg)=R_SUBQ) and is_shifter_const(taicpu(p).oper[2]^.val and taicpu(hp1).oper[2]^.val,OS_64)) or
- ((getsubreg(taicpu(p).oper[0]^.reg)=R_SUBL) and is_shifter_const(taicpu(p).oper[2]^.val and taicpu(hp1).oper[2]^.val,OS_32))
- ) or
- ((taicpu(p).oper[2]^.val and taicpu(hp1).oper[2]^.val)=0))
- {$endif AARCH64}
- then
- begin
- if not(RegUsedBetween(taicpu(hp1).oper[0]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole AndAnd2And done', p);
- AllocRegBetween(taicpu(hp1).oper[0]^.reg,p,hp1,UsedRegs);
- if (taicpu(p).oper[2]^.val and taicpu(hp1).oper[2]^.val)=0 then
- begin
- DebugMsg('Peephole AndAnd2Mov0 1 done', p);
- taicpu(p).opcode:=A_MOV;
- taicpu(p).ops:=2;
- taicpu(p).loadConst(1,0);
- taicpu(p).oppostfix:=taicpu(hp1).oppostfix;
- end
- else
- begin
- DebugMsg('Peephole AndAnd2And 1 done', p);
- taicpu(p).loadConst(2,taicpu(p).oper[2]^.val and taicpu(hp1).oper[2]^.val);
- taicpu(p).oppostfix:=taicpu(hp1).oppostfix;
- taicpu(p).loadReg(0,taicpu(hp1).oper[0]^.reg);
- end;
- asml.remove(hp1);
- hp1.free;
- Result:=true;
- exit;
- end
- else if not(RegUsedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- if (taicpu(p).oper[2]^.val and taicpu(hp1).oper[2]^.val)=0 then
- begin
- DebugMsg('Peephole AndAnd2Mov0 2 done', hp1);
- taicpu(hp1).opcode:=A_MOV;
- taicpu(hp1).loadConst(1,0);
- taicpu(hp1).ops:=2;
- taicpu(hp1).oppostfix:=taicpu(p).oppostfix;
- end
- else
- begin
- DebugMsg('Peephole AndAnd2And 2 done', hp1);
- AllocRegBetween(taicpu(p).oper[1]^.reg,p,hp1,UsedRegs);
- taicpu(hp1).loadConst(2,taicpu(p).oper[2]^.val and taicpu(hp1).oper[2]^.val);
- taicpu(hp1).oppostfix:=taicpu(p).oppostfix;
- taicpu(hp1).loadReg(1,taicpu(p).oper[1]^.reg);
- end;
- GetNextInstruction(p, hp1);
- RemoveCurrentP(p);
- p:=hp1;
- Result:=true;
- exit;
- end;
- end
- {
- change
- and reg2,reg1,$xxxxxxFF
- strb reg2,[...]
- dealloc reg2
- to
- strb reg1,[...]
- }
- else if ((taicpu(p).oper[2]^.val and $FF) = $FF) and
- MatchInstruction(p, A_AND, [C_None], [PF_None]) and
- GetNextInstructionUsingReg(p,hp1,taicpu(p).oper[0]^.reg) and
- MatchInstruction(hp1, A_STR, [C_None], [PF_B]) and
- assigned(FindRegDealloc(taicpu(p).oper[0]^.reg,tai(hp1.Next))) and
- { the reference in strb might not use reg2 }
- not(RegInRef(taicpu(p).oper[0]^.reg,taicpu(hp1).oper[1]^.ref^)) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole AndStrb2Strb done', p);
- {$ifdef AARCH64}
- taicpu(hp1).loadReg(0,newreg(R_INTREGISTER,getsupreg(taicpu(p).oper[1]^.reg),R_SUBD));
- {$else AARCH64}
- taicpu(hp1).loadReg(0,taicpu(p).oper[1]^.reg);
- {$endif AARCH64}
- AllocRegBetween(taicpu(p).oper[1]^.reg,p,hp1,UsedRegs);
- RemoveCurrentP(p);
- result:=true;
- exit;
- end
- {
- change
- and reg2,reg1,255
- uxtb/uxth reg3,reg2
- dealloc reg2
- to
- and reg3,reg1,x
- }
- else if MatchInstruction(p, A_AND, [C_None], [PF_None]) and
- GetNextInstructionUsingReg(p,hp1,taicpu(p).oper[0]^.reg) and
- ((((taicpu(p).oper[2]^.val and $ffffff00)=0) and MatchInstruction(hp1, A_UXTB, [C_None], [PF_None])) or
- (((taicpu(p).oper[2]^.val and $ffff0000)=0) and MatchInstruction(hp1, A_UXTH, [C_None], [PF_None]))) and
- (taicpu(hp1).ops = 2) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole AndUxt2And done', p);
- taicpu(hp1).opcode:=A_AND;
- taicpu(hp1).ops:=3;
- taicpu(hp1).loadReg(1,taicpu(p).oper[1]^.reg);
- taicpu(hp1).loadconst(2,taicpu(p).oper[2]^.val);
- GetNextInstruction(p,hp1);
- asml.remove(p);
- p.Free;
- p:=hp1;
- result:=true;
- exit;
- end
- else if ((taicpu(p).oper[2]^.val and $ffffff80)=0) and
- MatchInstruction(p, A_AND, [C_None], [PF_None]) and
- GetNextInstructionUsingReg(p,hp1,taicpu(p).oper[0]^.reg) and
- MatchInstruction(hp1, [A_SXTB,A_SXTH], [C_None], [PF_None]) and
- (taicpu(hp1).ops = 2) and
- RegEndofLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- { reg1 might not be modified inbetween }
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg,p,hp1)) then
- begin
- DebugMsg('Peephole AndSxt2And done', p);
- taicpu(hp1).opcode:=A_AND;
- taicpu(hp1).ops:=3;
- taicpu(hp1).loadReg(1,taicpu(p).oper[1]^.reg);
- setsubreg(taicpu(hp1).oper[1]^.reg,getsubreg(taicpu(hp1).oper[0]^.reg));
- taicpu(hp1).loadconst(2,taicpu(p).oper[2]^.val);
- GetNextInstruction(p,hp1);
- asml.remove(p);
- p.Free;
- p:=hp1;
- result:=true;
- exit;
- end
- {
- from
- and reg1,reg0,2^n-1
- mov reg2,reg1, lsl imm1
- (mov reg3,reg2, lsr/asr imm1)
- remove either the and or the lsl/xsr sequence if possible
- }
- else if (taicpu(p).oper[2]^.val < high(int64)) and
- cutils.ispowerof2(taicpu(p).oper[2]^.val+1,i) and
- GetNextInstructionUsingReg(p,hp1,taicpu(p).oper[0]^.reg) and
- MatchInstruction(hp1, A_MOV, [taicpu(p).condition], [PF_None]) and
- (taicpu(hp1).ops=3) and
- MatchOperand(taicpu(hp1).oper[1]^, taicpu(p).oper[0]^.reg) and
- (taicpu(hp1).oper[2]^.typ = top_shifterop) and
- {$ifdef ARM}
- (taicpu(hp1).oper[2]^.shifterop^.rs = NR_NO) and
- {$endif ARM}
- (taicpu(hp1).oper[2]^.shifterop^.shiftmode=SM_LSL) and
- RegEndOfLife(taicpu(p).oper[0]^.reg,taicpu(hp1)) then
- begin
- {
- and reg1,reg0,2^n-1
- mov reg2,reg1, lsl imm1
- mov reg3,reg2, lsr/asr imm1
- =>
- and reg1,reg0,2^n-1
- if lsr and 2^n-1>=imm1 or asr and 2^n-1>imm1
- }
- if GetNextInstructionUsingReg(hp1,hp2,taicpu(p).oper[0]^.reg) and
- MatchInstruction(hp2, A_MOV, [taicpu(p).condition], [PF_None]) and
- (taicpu(hp2).ops=3) and
- MatchOperand(taicpu(hp2).oper[1]^, taicpu(hp1).oper[0]^.reg) and
- (taicpu(hp2).oper[2]^.typ = top_shifterop) and
- {$ifdef ARM}
- (taicpu(hp2).oper[2]^.shifterop^.rs = NR_NO) and
- {$endif ARM}
- (taicpu(hp2).oper[2]^.shifterop^.shiftmode in [SM_ASR,SM_LSR]) and
- (taicpu(hp1).oper[2]^.shifterop^.shiftimm=taicpu(hp2).oper[2]^.shifterop^.shiftimm) and
- RegEndOfLife(taicpu(hp1).oper[0]^.reg,taicpu(hp2)) and
- ((i<32-taicpu(hp1).oper[2]^.shifterop^.shiftimm) or
- ((i=32-taicpu(hp1).oper[2]^.shifterop^.shiftimm) and
- (taicpu(hp2).oper[2]^.shifterop^.shiftmode=SM_LSR))) then
- begin
- DebugMsg('Peephole AndLslXsr2And done', p);
- taicpu(p).oper[0]^.reg:=taicpu(hp2).oper[0]^.reg;
- asml.Remove(hp1);
- asml.Remove(hp2);
- hp1.free;
- hp2.free;
- result:=true;
- exit;
- end
- {
- and reg1,reg0,2^n-1
- mov reg2,reg1, lsl imm1
- =>
- mov reg2,reg0, lsl imm1
- if imm1>i
- }
- else if (i>32-taicpu(hp1).oper[2]^.shifterop^.shiftimm) and
- not(RegModifiedBetween(taicpu(p).oper[1]^.reg, p, hp1)) then
- begin
- DebugMsg('Peephole AndLsl2Lsl done', p);
- taicpu(hp1).oper[1]^.reg:=taicpu(p).oper[1]^.reg;
- GetNextInstruction(p, hp1);
- asml.Remove(p);
- p.free;
- p:=hp1;
- result:=true;
- exit;
- end
- end;
- end;
- {
- change
- and reg1, ...
- mov reg2, reg1
- to
- and reg2, ...
- }
- if GetNextInstructionUsingReg(p, hp1, taicpu(p).oper[0]^.reg) and
- (taicpu(p).ops>=3) and
- RemoveSuperfluousMove(p, hp1, 'DataMov2Data') then
- Result:=true;
- end;
- function TARMAsmOptimizer.OptPass2Bitwise(var p: tai): Boolean;
- var
- hp1, hp2: tai;
- WorkingReg: TRegister;
- begin
- Result := False;
- {
- change
- and/bic reg1, ...
- ...
- cmp reg1, #0
- b<ne/eq> @Lbl
- to
- ands/bics reg1, ...
- Also:
- and/bic reg1, ...
- ...
- cmp reg1, #0
- (reg1 end of life)
- b<ne/eq> @Lbl
- to
- tst reg1, ...
- or
- bics xzr, reg1, ... under AArch64
- For ARM, also include OR, EOR and ORN
- }
- if (taicpu(p).condition = C_None) and
- (taicpu(p).ops>=3) and
- GetNextInstructionUsingReg(p, hp1, taicpu(p).oper[0]^.reg) and
- MatchInstruction(hp1, A_CMP, [C_None], [PF_None]) and
- MatchOperand(taicpu(hp1).oper[1]^, 0) and
- {$ifdef AARCH64}
- (SuperRegistersEqual(taicpu(hp1).oper[0]^.reg, taicpu(p).oper[0]^.reg)) and
- (
- (getsubreg(taicpu(hp1).oper[0]^.reg) = getsubreg(taicpu(p).oper[0]^.reg))
- or
- (
- (taicpu(p).oper[2]^.typ = top_const) and
- (taicpu(p).oper[2]^.val >= 0) and
- (taicpu(p).oper[2]^.val <= $FFFFFFFF)
- )
- ) and
- {$else AARCH64}
- (taicpu(hp1).oper[0]^.reg = taicpu(p).oper[0]^.reg) and
- {$endif AARCH64}
- not RegModifiedBetween(NR_DEFAULTFLAGS, p, hp1) and
- GetNextInstruction(hp1, hp2) then
- begin
- if MatchInstruction(hp2, [A_B, A_CMP, A_CMN, A_TST{$ifndef AARCH64}, A_TEQ{$endif not AARCH64}], [C_EQ, C_NE], [PF_None]) then
- begin
- AllocRegBetween(NR_DEFAULTFLAGS, p, hp1, UsedRegs);
- WorkingReg := taicpu(p).oper[0]^.reg;
- if
- {$ifndef AARCH64}
- (taicpu(p).opcode = A_AND) and
- {$endif AARCH64}
- RegEndOfLife(WorkingReg, taicpu(hp1)) then
- begin
- {$ifdef AARCH64}
- if (taicpu(p).opcode <> A_AND) then
- begin
- setsupreg(taicpu(p).oper[0]^.reg, RS_XZR);
- taicpu(p).oppostfix := PF_S;
- DebugMsg(SPeepholeOptimization + 'BIC; CMP -> BICS ' + gas_regname(taicpu(p).oper[0]^.reg), p);
- end
- else
- {$endif AARCH64}
- begin
- taicpu(p).opcode := A_TST;
- taicpu(p).oppostfix := PF_None;
- taicpu(p).loadreg(0, taicpu(p).oper[1]^.reg);
- taicpu(p).loadoper(1, taicpu(p).oper[2]^);
- if (taicpu(p).ops = 4) then
- begin
- { Make sure any shifter operator is also transferred }
- taicpu(p).loadshifterop(2, taicpu(p).oper[3]^.shifterop^);
- taicpu(p).ops := 3;
- end
- else
- taicpu(p).ops := 2;
- DebugMsg(SPeepholeOptimization + 'AND; CMP -> TST', p);
- end;
- end
- else
- begin
- taicpu(p).oppostfix := PF_S;
- {$ifdef AARCH64}
- DebugMsg(SPeepholeOptimization + 'AND/BIC; CMP -> ANDS/BICS', p);
- {$else AARCH64}
- DebugMsg(SPeepholeOptimization + 'Bitwise; CMP -> Bitwise+S', p);
- {$endif AARCH64}
- end;
- RemoveInstruction(hp1);
- { If a temporary register was used for and/cmp before, we might be
- able to deallocate the register so it can be used for other
- optimisations later }
- if (taicpu(p).opcode = A_TST) and TryRemoveRegAlloc(WorkingReg, p, p) then
- ExcludeRegFromUsedRegs(WorkingReg, UsedRegs);
- Result := True;
- Exit;
- end
- else if
- (hp2.typ = ait_label) or
- { Conditional comparison instructions have already been covered }
- RegModifiedByInstruction(NR_DEFAULTFLAGS, hp2) then
- begin
- { The comparison is a null operation }
- if RegEndOfLife(taicpu(p).oper[0]^.reg, taicpu(hp1)) then
- begin
- DebugMsg(SPeepholeOptimization + 'Bitwise; CMP -> nop', p);
- RemoveInstruction(hp1);
- RemoveCurrentP(p);
- end
- else
- begin
- DebugMsg(SPeepholeOptimization + 'CMP/BIC -> nop', hp1);
- RemoveInstruction(hp1);
- end;
- Result := True;
- Exit;
- end;
- end;
- end;
- function TARMAsmOptimizer.OptPass2TST(var p: tai): Boolean;
- var
- hp1, hp2: tai;
- begin
- Result := False;
- if
- {$ifndef AARCH64}
- (taicpu(p).condition = C_None) and
- {$endif AARCH64}
- GetNextInstruction(p, hp1) and
- MatchInstruction(hp1, A_B, [C_EQ, C_NE], [PF_None]) and
- GetNextInstructionUsingReg(hp1, hp2, taicpu(p).oper[0]^.reg) then
- begin
- case taicpu(hp2).opcode of
- A_AND:
- { Change:
- tst r1,##
- (r2 not in use, or r2 = r1)
- b.c .Lbl
- ...
- and r2,r1,##
- Optimise to:
- ands r2,r1,##
- b.c .Lbl
- ...
- }
- if (taicpu(hp2).oppostfix in [PF_None, PF_S]) and
- {$ifndef AARCH64}
- (taicpu(hp2).condition = C_None) and
- {$endif AARCH64}
- (taicpu(hp2).ops = taicpu(p).ops + 1) and
- not RegInUsedRegs(taicpu(hp2).oper[0]^.reg, UsedRegs) and
- MatchOperand(taicpu(hp2).oper[1]^, taicpu(p).oper[0]^.reg) and
- MatchOperand(taicpu(hp2).oper[2]^, taicpu(p).oper[1]^) and
- (
- (taicpu(hp2).ops = 3) or
- MatchOperand(taicpu(hp2).oper[3]^, taicpu(p).oper[2]^)
- ) and
- (
- not (cs_opt_level3 in current_settings.optimizerswitches) or
- (
- { Make sure the target register isn't used in between }
- not RegUsedBetween(taicpu(hp2).oper[0]^.reg, hp1, hp2) and
- (
- { If the second operand is a register, make sure it isn't modified in between }
- (taicpu(p).oper[1]^.typ <> top_reg) or
- not RegModifiedBetween(taicpu(p).oper[1]^.reg, hp1, hp2)
- )
- )
- ) then
- begin
- AllocRegBetween(taicpu(hp2).oper[0]^.reg, p, hp2, UsedRegs);
- if (taicpu(hp2).oppostfix = PF_S) then
- AllocRegBetween(NR_DEFAULTFLAGS, p, hp2, UsedRegs);
- DebugMsg(SPeepholeOptimization + 'TST; B.c; AND -> ANDS; B.c (TstBcAnd2AndsBc)', p);
- taicpu(hp2).oppostfix := PF_S;
- Asml.Remove(hp2);
- Asml.InsertAfter(hp2, p);
- RemoveCurrentP(p, hp2);
- Result := True;
- Exit;
- end;
- A_TST:
- { Change:
- tst r1,##
- b.c .Lbl
- ... (flags not modified)
- tst r1,##
- Remove second tst
- }
- if
- {$ifndef AARCH64}
- (taicpu(hp2).condition = C_None) and
- {$endif AARCH64}
- (taicpu(hp2).ops = taicpu(p).ops) and
- MatchOperand(taicpu(hp2).oper[0]^, taicpu(p).oper[0]^.reg) and
- MatchOperand(taicpu(hp2).oper[1]^, taicpu(p).oper[1]^) and
- (
- (taicpu(hp2).ops = 2) or
- MatchOperand(taicpu(hp2).oper[2]^, taicpu(p).oper[2]^)
- ) and
- (
- not (cs_opt_level3 in current_settings.optimizerswitches) or
- (
- { Make sure the flags aren't modified in between }
- not RegModifiedBetween(NR_DEFAULTFLAGS, hp1, hp2) and
- (
- { If the second operand is a register, make sure it isn't modified in between }
- (taicpu(p).oper[1]^.typ <> top_reg) or
- not RegModifiedBetween(taicpu(p).oper[1]^.reg, hp1, hp2)
- )
- )
- ) then
- begin
- DebugMsg(SPeepholeOptimization + 'TST; B.c; TST -> TST; B.c (TstBcTst2TstBc)', p);
- AllocRegBetween(NR_DEFAULTFLAGS, hp1, hp2, UsedRegs);
- RemoveInstruction(hp2);
- Result := True;
- Exit;
- end;
- else
- ;
- end;
- end;
- end;
- end.
|