fpc/compiler/procinfo.pas
pierre d1f31fab15 Merge commits 42525 and 45891 that add
checks for fpu exceptions for arm and aarch64.
------------------------------------------------------------------------
r42525 | florian | 2019-07-28 21:06:36 +0000 (Sun, 28 Jul 2019) | 2 lines

+ software handling of exceptions on arm
* reworked software handling of exceptions so they can be check lazily
------------------------------------------------------------------------
--- Merging r42525 into '.':
U    compiler/arm/cgcpu.pas
U    compiler/arm/narmadd.pas
U    compiler/arm/narminl.pas
U    compiler/arm/narmmat.pas
U    compiler/ncgcal.pas
U    compiler/procinfo.pas
U    rtl/arm/arm.inc
--- Recording mergeinfo for merge of r42525 into '.':
 U   .
Summary of conflicts:
  Tree conflicts: 1
------------------------------------------------------------------------
r42891 | florian | 2019-09-01 17:26:11 +0000 (Sun, 01 Sep 2019) | 1 line

+ support for software floating point exception handling on AArch64 (-CE)
------------------------------------------------------------------------
--- Merging r42891 into '.':
U    compiler/aarch64/cgcpu.pas
U    compiler/aarch64/ncpuadd.pas
U    compiler/aarch64/ncpuinl.pas
U    compiler/aarch64/ncpumat.pas
U    rtl/aarch64/aarch64.inc
U    rtl/aarch64/math.inc
U    rtl/aarch64/mathu.inc
--- Recording mergeinfo for merge of r42891 into '.':
 G   .

git-svn-id: branches/fixes_3_2@46225 -
2020-08-04 10:30:50 +00:00

322 lines
10 KiB
ObjectPascal

{
Copyright (c) 1998-2002 by Florian Klaempfl
Information about the current procedure that is being compiled
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 procinfo;
{$i fpcdefs.inc}
interface
uses
{ common }
cclasses,
{ global }
globtype,
{ symtable }
symconst,symtype,symdef,symsym,
node,
{ aasm }
cpubase,cgbase,cgutils,
aasmbase,aasmdata;
const
inherited_inlining_flags : tprocinfoflags =
[pi_do_call,
{ the stack frame can't be removed in this case }
pi_has_assembler_block,
pi_uses_exceptions];
type
tsavedlabels = array[Boolean] of TAsmLabel;
{ This object gives information on the current routine being
compiled.
}
{ tprocinfo }
tprocinfo = class(tlinkedlistitem)
private
{ list to store the procinfo's of the nested procedures }
nestedprocs : tlinkedlist;
{ required alignment for this stackframe }
fstackalignment : longint;
procedure addnestedproc(child: tprocinfo);
public
{ pointer to parent in nested procedures }
parent : tprocinfo;
{ the definition of the routine itself }
procdef : tprocdef;
{ nested implicit finalzation procedure, used for platform-specific
exception handling }
finalize_procinfo : tprocinfo;
{ file location of begin of procedure }
entrypos : tfileposinfo;
{ file location of end of procedure }
exitpos : tfileposinfo;
{ local switches at begin of procedure }
entryswitches : tlocalswitches;
{ local switches at end of procedure }
exitswitches : tlocalswitches;
{ Size of the parameters on the stack }
para_stack_size : pint;
{ Offset of temp after para/local are allocated }
tempstart : longint;
{ some collected informations about the procedure
see pi_xxxx constants above
}
flags : tprocinfoflags;
{ register used as frame pointer }
framepointer : tregister;
{ register containing currently the got }
got : tregister;
CurrGOTLabel : tasmlabel;
{ Holds the reference used to store all saved registers. }
save_regs_ref : treference;
{ Last assembler instruction of procedure prologue }
endprologue_ai : tlinkedlistitem;
{ Amount of stack adjustment after all alignments }
final_localsize : longint;
{ Labels for TRUE/FALSE condition, BREAK and CONTINUE }
CurrBreakLabel,
CurrContinueLabel : tasmlabel;
{ label to leave the sub routine }
CurrExitLabel : tasmlabel;
{ label for nested exits }
nestedexitlabel : tlabelsym;
{ The code for the routine itself, excluding entry and
exit code. This is a linked list of tai classes.
}
aktproccode : TAsmList;
{ Data (like jump tables) that belongs to this routine }
aktlocaldata : TAsmList;
{ max. of space need for parameters }
maxpushedparasize : aint;
{ some architectures need to know a stack size before the first compilation pass
estimatedtempsize contains an estimated value how big temps will get }
estimatedtempsize : longint;
{ is this a constructor that calls another constructor on itself
(either inherited, or another constructor of the same class)?
Requires different entry code for some targets. }
ConstructorCallingConstructor: boolean;
{ true, if an FPU instruction has been generated which could raise an exception and where the flags
need to be checked explicitly like on RISC-V or certain ARM architectures }
FPUExceptionCheckNeeded : Boolean;
constructor create(aparent:tprocinfo);virtual;
destructor destroy;override;
procedure allocate_push_parasize(size:longint);
function calc_stackframe_size:longint;virtual;abstract;
{ Set the address of the first temp, can be used to allocate
space for pushing parameters }
procedure set_first_temp_offset;virtual;
{ Generate parameter information }
procedure generate_parameter_info;virtual;
{ Allocate got register }
procedure allocate_got_register(list: TAsmList);virtual;
{ get frame pointer }
procedure init_framepointer; virtual;
{ Destroy the entire procinfo tree, starting from the outermost parent }
procedure destroy_tree;
function get_first_nestedproc: tprocinfo;
function has_nestedprocs: boolean;
function get_normal_proc: tprocinfo;
function create_for_outlining(const basesymname: string; astruct: tabstractrecorddef; potype: tproctypeoption; resultdef: tdef; entrynodeinfo: tnode): tprocinfo;
{ Add to parent's list of nested procedures even if parent is a 'main' procedure }
procedure force_nested;
{ Get the required alignment for the current stack frame }
property stackalignment: longint read fstackalignment;
{ Update the resuired alignment for the current stack frame based
on the current value and the new required alignment }
procedure updatestackalignment(alignment: longint);
{ Specific actions after the code has been generated }
procedure postprocess_code; virtual;
end;
tcprocinfo = class of tprocinfo;
var
cprocinfo : tcprocinfo;
{ information about the current sub routine being parsed (@var(pprocinfo))}
current_procinfo : tprocinfo;
implementation
uses
globals,cutils,systems,
procdefutil;
{****************************************************************************
TProcInfo
****************************************************************************}
constructor tprocinfo.create(aparent:tprocinfo);
begin
parent:=aparent;
procdef:=nil;
para_stack_size:=0;
fstackalignment:=target_info.stackalign;
flags:=[];
init_framepointer;
framepointer:=NR_FRAME_POINTER_REG;
maxpushedparasize:=0;
{ asmlists }
aktproccode:=TAsmList.Create;
aktlocaldata:=TAsmList.Create;
reference_reset(save_regs_ref,sizeof(aint),[]);
{ labels }
current_asmdata.getjumplabel(CurrExitLabel);
current_asmdata.getjumplabel(CurrGOTLabel);
CurrBreakLabel:=nil;
CurrContinueLabel:=nil;
if Assigned(parent) and (parent.procdef.parast.symtablelevel>=normal_function_level) then
parent.addnestedproc(Self);
end;
procedure tprocinfo.force_nested;
begin
if Assigned(parent) and (parent.procdef.parast.symtablelevel<normal_function_level) then
parent.addnestedproc(Self);
end;
destructor tprocinfo.destroy;
begin
nestedprocs.free;
aktproccode.free;
aktlocaldata.free;
end;
procedure tprocinfo.destroy_tree;
var
hp: tprocinfo;
begin
hp:=Self;
while Assigned(hp.parent) do
hp:=hp.parent;
hp.Free;
end;
procedure tprocinfo.addnestedproc(child: tprocinfo);
begin
if nestedprocs=nil then
nestedprocs:=TLinkedList.Create;
nestedprocs.insert(child);
end;
procedure tprocinfo.updatestackalignment(alignment: longint);
begin
fstackalignment:=max(fstackalignment,alignment);
end;
function tprocinfo.get_first_nestedproc: tprocinfo;
begin
if assigned(nestedprocs) then
result:=tprocinfo(nestedprocs.first)
else
result:=nil;
end;
function tprocinfo.has_nestedprocs: boolean;
begin
result:=assigned(nestedprocs) and (nestedprocs.count>0);
end;
function tprocinfo.get_normal_proc: tprocinfo;
begin
result:=self;
while assigned(result.parent) and (result.procdef.parast.symtablelevel>normal_function_level) do
result:=result.parent;
end;
function tprocinfo.create_for_outlining(const basesymname: string; astruct: tabstractrecorddef; potype: tproctypeoption; resultdef: tdef; entrynodeinfo: tnode): tprocinfo;
begin
result:=cprocinfo.create(self);
result.force_nested;
result.procdef:=create_outline_procdef(basesymname,astruct,potype,resultdef);
result.entrypos:=entrynodeinfo.fileinfo;
result.entryswitches:=entrynodeinfo.localswitches;
result.exitpos:=current_filepos; // filepos of last node?
result.exitswitches:=current_settings.localswitches; // localswitches of last node?
end;
procedure tprocinfo.allocate_push_parasize(size:longint);
begin
if size>maxpushedparasize then
maxpushedparasize:=size;
end;
procedure tprocinfo.set_first_temp_offset;
begin
end;
procedure tprocinfo.generate_parameter_info;
begin
{ generate callee paraloc register info, it initialises the size that
is allocated on the stack }
procdef.init_paraloc_info(calleeside);
para_stack_size:=procdef.calleeargareasize;
end;
procedure tprocinfo.allocate_got_register(list: TAsmList);
begin
{ most os/cpu combo's don't use this yet, so not yet abstract }
end;
procedure tprocinfo.init_framepointer;
begin
{ most targets use a constant, but some have a typed constant that must
be initialized }
end;
procedure tprocinfo.postprocess_code;
begin
{ no action by default }
end;
end.