| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114 | {    Copyright (c) 1998-2016 by Florian Klaempfl and Jonas Maebe    This unit contains helper procedures for the assembler peephole optimizer    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 aoptutils;{$i fpcdefs.inc}  interface    uses      cpubase,aasmtai,aasmcpu;    function MatchOpType(const p : taicpu;type0: toptype) : Boolean;    function MatchOpType(const p : taicpu;type0,type1 : toptype) : Boolean;{$if max_operands>2}    function MatchOpType(const p : taicpu; type0,type1,type2 : toptype) : Boolean;{$endif max_operands>2}    { skips all alignment fields and returns the next label (or non-align).      returns immediately with true if hp is a label }    function SkipAligns(hp: tai; out hp2: tai): boolean;    { skips all labels and returns the next "real" instruction }    function SkipLabels(hp: tai; out hp2: tai): boolean;    { sets hp2 to hp and returns True if hp is not nil }    function SetAndTest(const hp: tai; out hp2: tai): Boolean;  implementation    uses      aasmbase;    function MatchOpType(const p : taicpu; type0: toptype) : Boolean; inline;      begin        Result:=(p.ops=1) and (p.oper[0]^.typ=type0);      end;    function MatchOpType(const p : taicpu; type0,type1 : toptype) : Boolean; inline;      begin        Result:=(p.ops=2) and (p.oper[0]^.typ=type0) and (p.oper[1]^.typ=type1);      end;{$if max_operands>2}    function MatchOpType(const p : taicpu; type0,type1,type2 : toptype) : Boolean; inline;      begin        Result:=(p.ops=3) and (p.oper[0]^.typ=type0) and (p.oper[1]^.typ=type1) and (p.oper[2]^.typ=type2);      end;{$endif max_operands>2}    { skips all alignment fields and returns the next label (or non-align).      Returns immediately with True if hp is a label }    function SkipAligns(hp: tai; out hp2: tai): boolean;      begin        while assigned(hp) and              (hp.typ in SkipInstr + [ait_label,ait_align]) Do          begin            { Check that the label is actually live }            if (hp.typ = ait_label) and tai_label(hp).labsym.is_used then              Break;            hp := tai(hp.next);          end;        SkipAligns := SetAndTest(hp, hp2);      end;    { skips all labels and returns the next "real" instruction }    function SkipLabels(hp: tai; out hp2: tai): boolean;      begin        while assigned(hp.next) and              (tai(hp.next).typ in SkipInstr + [ait_label,ait_align]) Do          hp := tai(hp.next);        if assigned(hp.next) then          begin            SkipLabels := True;            hp2 := tai(hp.next)          end        else          begin            hp2 := hp;            SkipLabels := False          end;      end;    { sets hp2 to hp and returns True if hp is not nil }    function SetAndTest(const hp: tai; out hp2: tai): Boolean; inline;      begin        hp2 := hp;        Result := Assigned(hp);      end;end.
 |