diff options
Diffstat (limited to 'compiler/aoptbase.pas')
-rw-r--r-- | compiler/aoptbase.pas | 257 |
1 files changed, 257 insertions, 0 deletions
diff --git a/compiler/aoptbase.pas b/compiler/aoptbase.pas new file mode 100644 index 0000000000..83ad7b34ed --- /dev/null +++ b/compiler/aoptbase.pas @@ -0,0 +1,257 @@ +{ + Copyright (c) 1998-2002 by Jonas Maebe, member of the Free Pascal + Development Team + + This unit contains the base of all optimizer related objects + + 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 aoptbase; + +{$i fpcdefs.inc} + + interface + + uses + aasmbase,aasmcpu,aasmtai, + cpubase, + cgbase, + cgutils; + + Type + { the number of tai objects processed by an optimizer object since the last + time a register was modified } + { size at each dimension depends on the registers of this type } + TInstrSinceLastMod = Array[tregistertype] of pbyte; + + { the TAopBase object implements the basic methods that most other } + { assembler optimizer objects require } + Type + TAoptBase = class + { processor independent methods } + + constructor create; virtual; + destructor destroy;override; + { returns true if register Reg is used by instruction p1 } + Function RegInInstruction(Reg: TRegister; p1: tai): Boolean; + { returns true if register Reg occurs in operand op } + Function RegInOp(Reg: TRegister; const op: toper): Boolean; + { returns true if register Reg is used in the reference Ref } + Function RegInRef(Reg: TRegister; Const Ref: TReference): Boolean; + + { returns true if the references are completely equal } + {Function RefsEqual(Const R1, R2: TReference): Boolean;} + + { gets the next tai object after current that contains info relevant } + { to the optimizer in p1. If there is none, it returns false and } + { sets p1 to nil } + Function GetNextInstruction(Current: tai; Var Next: tai): Boolean; + { gets the previous tai object after current that contains info } + { relevant to the optimizer in last. If there is none, it retuns } + { false and sets last to nil } + Function GetLastInstruction(Current: tai; Var Last: tai): Boolean; + + + { processor dependent methods } + + { returns the maximum width component of Reg. Only has to be } + { overridden for the 80x86 (afaik) } + Function RegMaxSize(Reg: TRegister): TRegister; Virtual; + { returns true if Reg1 and Reg2 are of the samae width. Only has to } + { overridden for the 80x86 (afaik) } + Function RegsSameSize(Reg1, Reg2: TRegister): Boolean; Virtual; + { returns whether P is a load instruction (load contents from a } + { memory location or (register) variable into a register) } + Function IsLoadMemReg(p: tai): Boolean; Virtual; + { returns whether P is a load constant instruction (load a constant } + { into a register) } + Function IsLoadConstReg(p: tai): Boolean; Virtual; + { returns whether P is a store instruction (store contents from a } + { register to a memory location or to a (register) variable) } + Function IsStoreRegMem(p: tai): Boolean; Virtual; + + { create a paicpu Object that loads the contents of reg1 into reg2 } + Function a_load_reg_reg(reg1, reg2: TRegister): taicpu; Virtual; + + end; + + + implementation + + uses + globtype,globals, aoptcpub; + + constructor taoptbase.create; + begin + inherited create; + end; + + + destructor taoptbase.destroy; + begin + inherited destroy; + end; + + + Function TAOptBase.RegInInstruction(Reg: TRegister; p1: tai): Boolean; + Var Count: AWord; + TmpResult: Boolean; + Begin + TmpResult := False; + Count := 0; + If (p1.typ = ait_instruction) Then + Repeat + TmpResult := RegInOp(Reg, PInstr(p1)^.oper[Count]^); + Inc(Count) + Until (Count = MaxOps) or TmpResult; + RegInInstruction := TmpResult + End; + + + Function TAOptBase.RegInOp(Reg: TRegister; const op: toper): Boolean; + Begin + Case op.typ Of + Top_Reg: RegInOp := Reg = op.reg; + Top_Ref: RegInOp := RegInRef(Reg, op.ref^) + Else RegInOp := False + End + End; + + + Function TAOptBase.RegInRef(Reg: TRegister; Const Ref: TReference): Boolean; + Begin + Reg := RegMaxSize(Reg); + RegInRef := (Ref.Base = Reg) + {$ifdef RefsHaveIndexReg} + Or (Ref.Index = Reg) + {$endif RefsHaveIndexReg} + End; + + Function TAOptBase.GetNextInstruction(Current: tai; Var Next: tai): Boolean; + Begin + Repeat + Current := tai(Current.Next); + While Assigned(Current) And + ((Current.typ In SkipInstr) or +{$ifdef SPARC} + ((Current.typ=ait_instruction) and + (taicpu(Current).opcode=A_NOP) + ) or +{$endif SPARC} + ((Current.typ = ait_label) And + Not(Tai_Label(Current).l.is_used))) Do + Current := tai(Current.Next); + If Assigned(Current) And + (Current.typ = ait_Marker) And + (Tai_Marker(Current).Kind = NoPropInfoStart) Then + Begin + While Assigned(Current) And + ((Current.typ <> ait_Marker) Or + (Tai_Marker(Current).Kind <> NoPropInfoEnd)) Do + Current := Tai(Current.Next); + End; + Until Not(Assigned(Current)) Or + (Current.typ <> ait_Marker) Or + (Tai_Marker(Current).Kind <> NoPropInfoEnd); + Next := Current; + If Assigned(Current) And + Not((Current.typ In SkipInstr) or + ((Current.typ = ait_label) And + Not(Tai_Label(Current).l.is_used))) + Then GetNextInstruction := True + Else + Begin + Next := Nil; + GetNextInstruction := False; + End; + End; + + Function TAOptBase.GetLastInstruction(Current: tai; Var Last: tai): Boolean; + Begin + Repeat + Current := Tai(Current.previous); + While Assigned(Current) And + (((Current.typ = ait_Marker) And + Not(Tai_Marker(Current).Kind in [AsmBlockEnd,NoPropInfoEnd])) or + (Current.typ In SkipInstr) or + ((Current.typ = ait_label) And + Not(Tai_Label(Current).l.is_used))) Do + Current := Tai(Current.previous); + If Assigned(Current) And + (Current.typ = ait_Marker) And + (Tai_Marker(Current).Kind = NoPropInfoEnd) Then + Begin + While Assigned(Current) And + ((Current.typ <> ait_Marker) Or + (Tai_Marker(Current).Kind <> NoPropInfoStart)) Do + Current := Tai(Current.previous); + End; + Until Not(Assigned(Current)) Or + (Current.typ <> ait_Marker) Or + (Tai_Marker(Current).Kind <> NoPropInfoStart); + If Not(Assigned(Current)) or + (Current.typ In SkipInstr) or + ((Current.typ = ait_label) And + Not(Tai_Label(Current).l.is_used)) or + ((Current.typ = ait_Marker) And + (Tai_Marker(Current).Kind = AsmBlockEnd)) + Then + Begin + Last := Nil; + GetLastInstruction := False + End + Else + Begin + Last := Current; + GetLastInstruction := True; + End; + End; + + + { ******************* Processor dependent stuff *************************** } + + Function TAOptBase.RegMaxSize(Reg: TRegister): TRegister; + Begin + RegMaxSize := Reg + End; + + Function TAOptBase.RegsSameSize(Reg1, Reg2: TRegister): Boolean; + Begin + RegsSameSize := True + End; + + Function TAOptBase.IsLoadMemReg(p: tai): Boolean; + Begin + Abstract + End; + + Function TAOptBase.IsLoadConstReg(p: tai): Boolean; + Begin + Abstract + End; + + Function TAOptBase.IsStoreRegMem(p: tai): Boolean; + Begin + Abstract + End; + + Function TAoptBase.a_load_reg_reg(reg1, reg2: TRegister): taicpu; + Begin + Abstract + End; + +end. |