fpc/compiler/aarch64/cpupi.pas
svenbarth 2b59000d56 + implement compiler support for SEH on Win64
Note: due to the way we access variables in a nested function (which in this case includes exception filters) we can not extract the finally handlers and call them (like we do on i386 and x86_64, but instead we duplicate the finally code)

git-svn-id: trunk@44941 -
2020-04-21 06:06:05 +00:00

148 lines
4.6 KiB
ObjectPascal

{
Copyright (c) 2002 by Florian Klaempfl
This unit contains the CPU specific part of tprocinfo
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 cpupi;
{$i fpcdefs.inc}
interface
uses
procinfo,
psub,
aasmdata,aasmbase;
type
tcpuprocinfo=class(tcgprocinfo)
private
scopes: TAsmList;
scopecount: longint;
unwindflags: byte;
public
constructor create(aparent: tprocinfo); override;
destructor destroy; override;
procedure set_first_temp_offset; override;
procedure add_finally_scope(startlabel,endlabel,handler:TAsmSymbol;implicit:Boolean);
procedure add_except_scope(trylabel,exceptlabel,endlabel,filter:TAsmSymbol);
procedure dump_scopes(list:tasmlist);
end;
implementation
uses
cutils,
fmodule,
symtable,
tgobj,
cpubase,
aasmtai;
const
SCOPE_FINALLY=0;
SCOPE_CATCHALL=1;
SCOPE_IMPLICIT=2;
constructor tcpuprocinfo.create(aparent: tprocinfo);
begin
inherited;
{ use the stack pointer as framepointer, because
1) we exactly know the offsets of the temps from the stack pointer
after pass 1 (based on the require parameter stack size for called
routines), while we don't know it for the frame pointer (it depends
on the number of saved registers)
2) temp offsets from the stack pointer are positive while those from
the frame pointer are negative, and we can directly encode much
bigger positive offsets in the instructions
}
framepointer:=NR_STACK_POINTER_REG;
end;
destructor tcpuprocinfo.destroy;
begin
scopes.free;
inherited destroy;
end;
procedure tcpuprocinfo.set_first_temp_offset;
begin
{ leave room for allocated parameters }
tg.setfirsttemp(align(maxpushedparasize,16));
end;
procedure tcpuprocinfo.add_finally_scope(startlabel,endlabel,handler:TAsmSymbol;implicit:Boolean);
begin
unwindflags:=unwindflags or 2;
if implicit then { also needs catch functionality }
unwindflags:=unwindflags or 1;
inc(scopecount);
if scopes=nil then
scopes:=TAsmList.Create;
if implicit then
scopes.concat(tai_const.create_32bit(SCOPE_IMPLICIT))
else
scopes.concat(tai_const.create_32bit(SCOPE_FINALLY));
scopes.concat(tai_const.create_rva_sym(startlabel));
scopes.concat(tai_const.create_rva_sym(endlabel));
scopes.concat(tai_const.create_rva_sym(handler));
end;
procedure tcpuprocinfo.add_except_scope(trylabel,exceptlabel,endlabel,filter:TAsmSymbol);
begin
unwindflags:=unwindflags or 3;
inc(scopecount);
if scopes=nil then
scopes:=TAsmList.Create;
if Assigned(filter) then
scopes.concat(tai_const.create_rva_sym(filter))
else
scopes.concat(tai_const.create_32bit(SCOPE_CATCHALL));
scopes.concat(tai_const.create_rva_sym(trylabel));
scopes.concat(tai_const.create_rva_sym(exceptlabel));
scopes.concat(tai_const.create_rva_sym(endlabel));
end;
procedure tcpuprocinfo.dump_scopes(list: tasmlist);
var
hdir: tai_seh_directive;
begin
if (scopecount=0) then
exit;
hdir:=cai_seh_directive.create_name(ash_handler,'__FPC_specific_handler');
if not systemunit.iscurrentunit then
current_module.add_extern_asmsym('__FPC_specific_handler',AB_EXTERNAL,AT_FUNCTION);
hdir.data.flags:=unwindflags;
list.concat(hdir);
list.concat(cai_seh_directive.create(ash_handlerdata));
inc(list.section_count);
list.concat(tai_const.create_32bit(scopecount));
list.concatlist(scopes);
{ return to text, required for GAS compatibility }
{ This creates a tai_align which is redundant here (although harmless) }
new_section(list,sec_code,lower(procdef.mangledname),0);
end;
begin
cprocinfo:=tcpuprocinfo;
end.