{ Copyright (c) 1998-2002 by Jonas Maebe, member of the Free Pascal Development Team This unit implements the common RiscV optimizer object 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 aoptcpurv; interface {$I fpcdefs.inc} {$ifdef EXTDEBUG} {$define DEBUG_AOPTCPU} {$endif EXTDEBUG} uses cpubase, globals, globtype, cgbase, aoptobj, aoptcpub, aopt, aasmtai, aasmcpu; type TRVCpuAsmOptimizer = class(TAsmOptimizer) function InstructionLoadsFromReg(const reg: TRegister; const hp: tai): boolean; override; function RegLoadedWithNewValue(reg: tregister; hp: tai): boolean; override; function RegModifiedByInstruction(Reg: TRegister; p1: tai): boolean; override; Function GetNextInstructionUsingReg(Current: tai; Out Next: tai; reg: TRegister): Boolean; { outputs a debug message into the assembler file } procedure DebugMsg(const s: string; p: tai); function PeepHoleOptPass1Cpu(var p: tai): boolean; override; function OptPass1OP(var p: tai): boolean; function OptPass1FOP(var p: tai;mvop: tasmop): boolean; function OptPass1FSGNJ(var p: tai;mvop: tasmop): boolean; function OptPass1SLTx(var p: tai): boolean; function OptPass1SLTI(var p: tai): boolean; function OptPass1Andi(var p: tai): boolean; function OptPass1SLTIU(var p: tai): boolean; function OptPass1SxxI(var p: tai): boolean; function OptPass1Add(var p: tai): boolean; function OptPass1Sub(var p: tai): boolean; function OptPass1Fcmp(var p: tai): boolean; procedure RemoveInstr(var orig: tai; moveback: boolean=true); end; implementation uses cutils, verbose; function MatchInstruction(const instr: tai; const op: TCommonAsmOps; const AConditions: TAsmConds = []): boolean; begin result := (instr.typ = ait_instruction) and (taicpu(instr).opcode in op) and ((AConditions=[]) or (taicpu(instr).condition in AConditions)); end; function MatchInstruction(const instr: tai; const op: TAsmOp; const AConditions: TAsmConds = []): boolean; begin result := (instr.typ = ait_instruction) and (taicpu(instr).opcode = op) and ((AConditions=[]) or (taicpu(instr).condition in AConditions)); 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_ref: Result:=RefsEqual(oper1.ref^, oper2.ref^);} else Result:=false; end end; function MatchOperand(const oper: TOper; const reg: TRegister): boolean; inline; begin result := (oper.typ = top_reg) and (oper.reg = reg); end; {$ifdef DEBUG_AOPTCPU} procedure TRVCpuAsmOptimizer.DebugMsg(const s: string;p : tai); begin asml.insertbefore(tai_comment.Create(strpnew(s)), p); end; {$else DEBUG_AOPTCPU} procedure TRVCpuAsmOptimizer.DebugMsg(const s: string;p : tai);inline; begin end; {$endif DEBUG_AOPTCPU} function TRVCpuAsmOptimizer.InstructionLoadsFromReg(const reg: TRegister; const hp: tai): boolean; var p: taicpu; i: longint; begin result:=false; if not (assigned(hp) and (hp.typ=ait_instruction)) then exit; p:=taicpu(hp); i:=0; while(i