mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-05 15:47:54 +02:00
565 lines
19 KiB
ObjectPascal
565 lines
19 KiB
ObjectPascal
{
|
|
Copyright (c) 2011-2020 by Free Pascal development team
|
|
|
|
Generate Win64-specific exception handling code (based on x86_64 code)
|
|
|
|
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 ncpuflw;
|
|
|
|
{$i fpcdefs.inc}
|
|
|
|
interface
|
|
|
|
uses
|
|
node,nflw,ncgflw,psub;
|
|
|
|
type
|
|
taarch64raisenode=class(tcgraisenode)
|
|
function pass_1 : tnode;override;
|
|
end;
|
|
|
|
taarch64onnode=class(tcgonnode)
|
|
procedure pass_generate_code;override;
|
|
end;
|
|
|
|
taarch64tryexceptnode=class(tcgtryexceptnode)
|
|
procedure pass_generate_code;override;
|
|
end;
|
|
|
|
taarch64tryfinallynode=class(tcgtryfinallynode)
|
|
finalizepi: tcgprocinfo;
|
|
constructor create(l,r:TNode);override;
|
|
constructor create_implicit(l,r:TNode);override;
|
|
function simplify(forinline: boolean): tnode;override;
|
|
procedure pass_generate_code;override;
|
|
function dogetcopy:tnode;override;
|
|
end;
|
|
|
|
implementation
|
|
|
|
uses
|
|
globtype,globals,verbose,systems,fmodule,
|
|
nbas,ncal,nutils,
|
|
symconst,symsym,symdef,
|
|
cgbase,cgobj,cgutils,tgobj,
|
|
cpubase,htypechk,
|
|
pass_1,pass_2,
|
|
aasmbase,aasmtai,aasmdata,aasmcpu,procinfo,cpupi;
|
|
|
|
var
|
|
endexceptlabel: tasmlabel;
|
|
|
|
|
|
{ taarch64raisenode }
|
|
|
|
function taarch64raisenode.pass_1 : tnode;
|
|
var
|
|
statements : tstatementnode;
|
|
raisenode : tcallnode;
|
|
begin
|
|
{ difference from generic code is that address stack is not popped on reraise }
|
|
if (target_info.system<>system_aarch64_win64) or assigned(left) then
|
|
result:=inherited pass_1
|
|
else
|
|
begin
|
|
result:=internalstatements(statements);
|
|
raisenode:=ccallnode.createintern('fpc_reraise',nil);
|
|
include(raisenode.callnodeflags,cnf_call_never_returns);
|
|
addstatement(statements,raisenode);
|
|
end;
|
|
end;
|
|
|
|
{ taarch64onnode }
|
|
|
|
procedure taarch64onnode.pass_generate_code;
|
|
var
|
|
exceptvarsym : tlocalvarsym;
|
|
begin
|
|
if (target_info.system<>system_aarch64_win64) then
|
|
begin
|
|
inherited pass_generate_code;
|
|
exit;
|
|
end;
|
|
|
|
location_reset(location,LOC_VOID,OS_NO);
|
|
|
|
{ RTL will put exceptobject into X0 when jumping here }
|
|
cg.a_reg_alloc(current_asmdata.CurrAsmList,NR_FUNCTION_RESULT_REG);
|
|
|
|
{ Retrieve exception variable }
|
|
if assigned(excepTSymtable) then
|
|
exceptvarsym:=tlocalvarsym(excepTSymtable.SymList[0])
|
|
else
|
|
exceptvarsym:=nil;
|
|
|
|
if assigned(exceptvarsym) then
|
|
begin
|
|
exceptvarsym.localloc.loc:=LOC_REFERENCE;
|
|
exceptvarsym.localloc.size:=OS_ADDR;
|
|
tg.GetLocal(current_asmdata.CurrAsmList,sizeof(pint),voidpointertype,exceptvarsym.localloc.reference);
|
|
cg.a_load_reg_ref(current_asmdata.CurrAsmList,OS_ADDR,OS_ADDR,NR_FUNCTION_RESULT_REG,exceptvarsym.localloc.reference);
|
|
end;
|
|
cg.a_reg_dealloc(current_asmdata.CurrAsmList,NR_FUNCTION_RESULT_REG);
|
|
|
|
if assigned(right) then
|
|
secondpass(right);
|
|
|
|
{ deallocate exception symbol }
|
|
if assigned(exceptvarsym) then
|
|
begin
|
|
tg.UngetLocal(current_asmdata.CurrAsmList,exceptvarsym.localloc.reference);
|
|
exceptvarsym.localloc.loc:=LOC_INVALID;
|
|
end;
|
|
cg.g_call(current_asmdata.CurrAsmList,'FPC_DONEEXCEPTION');
|
|
cg.a_jmp_always(current_asmdata.CurrAsmList,endexceptlabel);
|
|
end;
|
|
|
|
{ taarch64tryfinallynode }
|
|
|
|
function reset_regvars(var n: tnode; arg: pointer): foreachnoderesult;
|
|
begin
|
|
case n.nodetype of
|
|
temprefn:
|
|
make_not_regable(n,[]);
|
|
calln:
|
|
include(tprocinfo(arg).flags,pi_do_call);
|
|
else
|
|
;
|
|
end;
|
|
result:=fen_true;
|
|
end;
|
|
|
|
function copy_parasize(var n: tnode; arg: pointer): foreachnoderesult;
|
|
begin
|
|
case n.nodetype of
|
|
calln:
|
|
tcgprocinfo(arg).allocate_push_parasize(tcallnode(n).pushed_parasize);
|
|
else
|
|
;
|
|
end;
|
|
result:=fen_true;
|
|
end;
|
|
|
|
constructor taarch64tryfinallynode.create(l, r: TNode);
|
|
begin
|
|
inherited create(l,r);
|
|
if (target_info.system=system_aarch64_win64) and
|
|
{ Don't create child procedures for generic methods, their nested-like
|
|
behavior causes compilation errors because real nested procedures
|
|
aren't allowed for generics. Not creating them doesn't harm because
|
|
generic node tree is discarded without generating code. }
|
|
not (df_generic in current_procinfo.procdef.defoptions) then
|
|
begin
|
|
finalizepi:=tcgprocinfo(current_procinfo.create_for_outlining('$fin$',current_procinfo.procdef.struct,potype_exceptfilter,voidtype,r));
|
|
{ the init/final code is messing with asm nodes, so inform the compiler about this }
|
|
include(finalizepi.flags,pi_has_assembler_block);
|
|
{ Regvar optimization for symbols is suppressed when using exceptions, but
|
|
temps may be still placed into registers. This must be fixed. }
|
|
foreachnodestatic(r,@reset_regvars,finalizepi);
|
|
end;
|
|
end;
|
|
|
|
constructor taarch64tryfinallynode.create_implicit(l, r: TNode);
|
|
begin
|
|
inherited create_implicit(l, r);
|
|
if (target_info.system=system_aarch64_win64) then
|
|
begin
|
|
if df_generic in current_procinfo.procdef.defoptions then
|
|
InternalError(2020033101);
|
|
|
|
finalizepi:=tcgprocinfo(current_procinfo.create_for_outlining('$fin$',current_procinfo.procdef.struct,potype_exceptfilter,voidtype,r));
|
|
include(finalizepi.flags,pi_do_call);
|
|
{ the init/final code is messing with asm nodes, so inform the compiler about this }
|
|
include(finalizepi.flags,pi_has_assembler_block);
|
|
finalizepi.allocate_push_parasize(32);
|
|
end;
|
|
end;
|
|
|
|
function taarch64tryfinallynode.simplify(forinline: boolean): tnode;
|
|
begin
|
|
result:=inherited simplify(forinline);
|
|
if (target_info.system<>system_aarch64_win64) then
|
|
exit;
|
|
if (result=nil) then
|
|
begin
|
|
{ generate a copy of the code }
|
|
finalizepi.code:=right.getcopy;
|
|
foreachnodestatic(right,@copy_parasize,finalizepi);
|
|
{ For implicit frames, no actual code is available at this time,
|
|
it is added later in assembler form. So store the nested procinfo
|
|
for later use. }
|
|
if implicitframe then
|
|
begin
|
|
current_procinfo.finalize_procinfo:=finalizepi;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
procedure emit_nop;
|
|
var
|
|
dummy: TAsmLabel;
|
|
begin
|
|
{ To avoid optimizing away the whole thing, prepend a jumplabel with increased refcount }
|
|
current_asmdata.getjumplabel(dummy);
|
|
dummy.increfs;
|
|
cg.a_label(current_asmdata.CurrAsmList,dummy);
|
|
current_asmdata.CurrAsmList.concat(Taicpu.op_none(A_NOP));
|
|
end;
|
|
|
|
procedure taarch64tryfinallynode.pass_generate_code;
|
|
var
|
|
trylabel,
|
|
endtrylabel,
|
|
finallylabel,
|
|
endfinallylabel,
|
|
templabel,
|
|
oldexitlabel: tasmlabel;
|
|
oldflowcontrol: tflowcontrol;
|
|
catch_frame: boolean;
|
|
begin
|
|
if (target_info.system<>system_aarch64_win64) then
|
|
begin
|
|
inherited pass_generate_code;
|
|
exit;
|
|
end;
|
|
|
|
location_reset(location,LOC_VOID,OS_NO);
|
|
|
|
{ Do not generate a frame that catches exceptions if the only action
|
|
would be reraising it. Doing so is extremely inefficient with SEH
|
|
(in contrast with setjmp/longjmp exception handling) }
|
|
catch_frame:=implicitframe and
|
|
(current_procinfo.procdef.proccalloption=pocall_safecall);
|
|
|
|
oldflowcontrol:=flowcontrol;
|
|
flowcontrol:=[fc_inflowcontrol];
|
|
|
|
templabel:=nil;
|
|
current_asmdata.getjumplabel(trylabel);
|
|
current_asmdata.getjumplabel(endtrylabel);
|
|
current_asmdata.getjumplabel(finallylabel);
|
|
current_asmdata.getjumplabel(endfinallylabel);
|
|
oldexitlabel:=current_procinfo.CurrExitLabel;
|
|
if implicitframe then
|
|
current_procinfo.CurrExitLabel:=finallylabel;
|
|
|
|
{ Start of scope }
|
|
{ Padding with NOP is necessary here because exceptions in called
|
|
procedures are seen at the next instruction, while CPU/OS exceptions
|
|
like AV are seen at the current instruction.
|
|
|
|
So in the following code
|
|
|
|
raise_some_exception; //(a)
|
|
try
|
|
pchar(nil)^:='0'; //(b)
|
|
...
|
|
|
|
without NOP, exceptions (a) and (b) will be seen at the same address
|
|
and fall into the same scope. However they should be seen in different scopes.
|
|
}
|
|
|
|
emit_nop;
|
|
cg.a_label(current_asmdata.CurrAsmList,trylabel);
|
|
|
|
{ try code }
|
|
if assigned(left) then
|
|
begin
|
|
{ fc_unwind_xx tells exit/continue/break statements to emit special
|
|
unwind code instead of just JMP }
|
|
if not implicitframe then
|
|
flowcontrol:=flowcontrol+[fc_catching_exceptions,fc_unwind_exit,fc_unwind_loop];
|
|
secondpass(left);
|
|
flowcontrol:=flowcontrol-[fc_catching_exceptions,fc_unwind_exit,fc_unwind_loop];
|
|
if codegenerror then
|
|
exit;
|
|
end;
|
|
|
|
{ finallylabel is only used in implicit frames as an exit point from nested try..finally
|
|
statements, if any. To prevent finalizer from being executed twice, it must come before
|
|
endtrylabel (bug #34772) }
|
|
if catch_frame then
|
|
begin
|
|
current_asmdata.getjumplabel(templabel);
|
|
cg.a_label(current_asmdata.CurrAsmList, finallylabel);
|
|
{ jump over exception handler }
|
|
cg.a_jmp_always(current_asmdata.CurrAsmList,templabel);
|
|
{ Handle the except block first, so endtrylabel serves both
|
|
as end of scope and as unwind target. This way it is possible to
|
|
encode everything into a single scope record. }
|
|
cg.a_label(current_asmdata.CurrAsmList,endtrylabel);
|
|
if (current_procinfo.procdef.proccalloption=pocall_safecall) then
|
|
begin
|
|
handle_safecall_exception;
|
|
cg.a_jmp_always(current_asmdata.CurrAsmList,endfinallylabel);
|
|
end
|
|
else
|
|
InternalError(2014031601);
|
|
cg.a_label(current_asmdata.CurrAsmList,templabel);
|
|
end
|
|
else
|
|
begin
|
|
{ same as emit_nop but using finallylabel instead of dummy }
|
|
cg.a_label(current_asmdata.CurrAsmList,finallylabel);
|
|
finallylabel.increfs;
|
|
current_asmdata.CurrAsmList.concat(Taicpu.op_none(A_NOP));
|
|
cg.a_label(current_asmdata.CurrAsmList,endtrylabel);
|
|
end;
|
|
|
|
flowcontrol:=[fc_inflowcontrol];
|
|
{ store the tempflags so that we can generate a copy of the finally handler
|
|
later on }
|
|
if not implicitframe then
|
|
finalizepi.store_tempflags;
|
|
{ generate the inline finalizer code }
|
|
secondpass(right);
|
|
|
|
if codegenerror then
|
|
exit;
|
|
|
|
{ normal exit from safecall proc must zero the result register }
|
|
if implicitframe and (current_procinfo.procdef.proccalloption=pocall_safecall) then
|
|
cg.a_load_const_reg(current_asmdata.CurrAsmList,OS_INT,0,NR_FUNCTION_RESULT_REG);
|
|
|
|
cg.a_label(current_asmdata.CurrAsmList,endfinallylabel);
|
|
|
|
{ generate the scope record in .xdata }
|
|
tcpuprocinfo(current_procinfo).add_finally_scope(trylabel,endtrylabel,
|
|
current_asmdata.RefAsmSymbol(finalizepi.procdef.mangledname,AT_FUNCTION),catch_frame);
|
|
|
|
if implicitframe then
|
|
current_procinfo.CurrExitLabel:=oldexitlabel;
|
|
flowcontrol:=oldflowcontrol;
|
|
end;
|
|
|
|
function taarch64tryfinallynode.dogetcopy: tnode;
|
|
var
|
|
p : taarch64tryfinallynode absolute result;
|
|
begin
|
|
result:=inherited dogetcopy;
|
|
if (target_info.system=system_aarch64_win64) then
|
|
begin
|
|
if df_generic in current_procinfo.procdef.defoptions then
|
|
InternalError(2020033104);
|
|
|
|
p.finalizepi:=tcgprocinfo(current_procinfo.create_for_outlining('$fin$',current_procinfo.procdef.struct,potype_exceptfilter,voidtype,p.right));
|
|
if pi_do_call in finalizepi.flags then
|
|
include(p.finalizepi.flags,pi_do_call);
|
|
{ the init/final code is messing with asm nodes, so inform the compiler about this }
|
|
include(p.finalizepi.flags,pi_has_assembler_block);
|
|
if implicitframe then
|
|
p.finalizepi.allocate_push_parasize(32);
|
|
end;
|
|
end;
|
|
|
|
{ taarch64tryexceptnode }
|
|
|
|
procedure taarch64tryexceptnode.pass_generate_code;
|
|
var
|
|
trylabel,
|
|
exceptlabel,oldendexceptlabel,
|
|
lastonlabel,
|
|
exitexceptlabel,
|
|
continueexceptlabel,
|
|
breakexceptlabel,
|
|
oldCurrExitLabel,
|
|
oldContinueLabel,
|
|
oldBreakLabel : tasmlabel;
|
|
onlabel,
|
|
filterlabel: tasmlabel;
|
|
oldflowcontrol,tryflowcontrol,
|
|
exceptflowcontrol : tflowcontrol;
|
|
hnode : tnode;
|
|
hlist : tasmlist;
|
|
onnodecount : tai_const;
|
|
sym : tasmsymbol;
|
|
label
|
|
errorexit;
|
|
begin
|
|
if (target_info.system<>system_aarch64_win64) then
|
|
begin
|
|
inherited pass_generate_code;
|
|
exit;
|
|
end;
|
|
location_reset(location,LOC_VOID,OS_NO);
|
|
|
|
oldflowcontrol:=flowcontrol;
|
|
exceptflowcontrol:=[];
|
|
continueexceptlabel:=nil;
|
|
breakexceptlabel:=nil;
|
|
|
|
include(flowcontrol,fc_inflowcontrol);
|
|
{ this can be called recursivly }
|
|
oldBreakLabel:=nil;
|
|
oldContinueLabel:=nil;
|
|
oldendexceptlabel:=endexceptlabel;
|
|
|
|
{ save the old labels for control flow statements }
|
|
oldCurrExitLabel:=current_procinfo.CurrExitLabel;
|
|
current_asmdata.getjumplabel(exitexceptlabel);
|
|
if assigned(current_procinfo.CurrBreakLabel) then
|
|
begin
|
|
oldContinueLabel:=current_procinfo.CurrContinueLabel;
|
|
oldBreakLabel:=current_procinfo.CurrBreakLabel;
|
|
current_asmdata.getjumplabel(breakexceptlabel);
|
|
current_asmdata.getjumplabel(continueexceptlabel);
|
|
end;
|
|
|
|
current_asmdata.getjumplabel(exceptlabel);
|
|
current_asmdata.getjumplabel(endexceptlabel);
|
|
current_asmdata.getjumplabel(lastonlabel);
|
|
filterlabel:=nil;
|
|
|
|
{ start of scope }
|
|
current_asmdata.getjumplabel(trylabel);
|
|
emit_nop;
|
|
cg.a_label(current_asmdata.CurrAsmList,trylabel);
|
|
|
|
{ control flow in try block needs no special handling,
|
|
just make sure that target labels are outside the scope }
|
|
secondpass(left);
|
|
tryflowcontrol:=flowcontrol;
|
|
if codegenerror then
|
|
goto errorexit;
|
|
|
|
{ jump over except handlers }
|
|
cg.a_jmp_always(current_asmdata.CurrAsmList,endexceptlabel);
|
|
|
|
{ end of scope }
|
|
cg.a_label(current_asmdata.CurrAsmList,exceptlabel);
|
|
|
|
{ set control flow labels for the except block }
|
|
{ and the on statements }
|
|
current_procinfo.CurrExitLabel:=exitexceptlabel;
|
|
if assigned(oldBreakLabel) then
|
|
begin
|
|
current_procinfo.CurrContinueLabel:=continueexceptlabel;
|
|
current_procinfo.CurrBreakLabel:=breakexceptlabel;
|
|
end;
|
|
|
|
flowcontrol:=[fc_inflowcontrol];
|
|
{ on statements }
|
|
if assigned(right) then
|
|
begin
|
|
{ emit filter table to a temporary asmlist }
|
|
hlist:=TAsmList.Create;
|
|
current_asmdata.getaddrlabel(filterlabel);
|
|
new_section(hlist,sec_rodata_norel,filterlabel.name,4);
|
|
cg.a_label(hlist,filterlabel);
|
|
onnodecount:=tai_const.create_32bit(0);
|
|
hlist.concat(onnodecount);
|
|
|
|
hnode:=right;
|
|
while assigned(hnode) do
|
|
begin
|
|
if hnode.nodetype<>onn then
|
|
InternalError(2011103101);
|
|
current_asmdata.getjumplabel(onlabel);
|
|
sym:=current_asmdata.RefAsmSymbol(tonnode(hnode).excepttype.vmt_mangledname,AT_DATA,true);
|
|
hlist.concat(tai_const.create_rva_sym(sym));
|
|
hlist.concat(tai_const.create_rva_sym(onlabel));
|
|
current_module.add_extern_asmsym(sym);
|
|
cg.a_label(current_asmdata.CurrAsmList,onlabel);
|
|
secondpass(hnode);
|
|
inc(onnodecount.value);
|
|
hnode:=tonnode(hnode).left;
|
|
end;
|
|
{ add 'else' node to the filter list, too }
|
|
if assigned(t1) then
|
|
begin
|
|
hlist.concat(tai_const.create_32bit(-1));
|
|
hlist.concat(tai_const.create_rva_sym(lastonlabel));
|
|
inc(onnodecount.value);
|
|
end;
|
|
{ now move filter table to permanent list all at once }
|
|
current_procinfo.aktlocaldata.concatlist(hlist);
|
|
hlist.free;
|
|
end;
|
|
|
|
cg.a_label(current_asmdata.CurrAsmList,lastonlabel);
|
|
if assigned(t1) then
|
|
begin
|
|
{ here we don't have to reset flowcontrol }
|
|
{ the default and on flowcontrols are handled equal }
|
|
secondpass(t1);
|
|
cg.g_call(current_asmdata.CurrAsmList,'FPC_DONEEXCEPTION');
|
|
if (flowcontrol*[fc_exit,fc_break,fc_continue]<>[]) then
|
|
cg.a_jmp_always(current_asmdata.CurrAsmList,endexceptlabel);
|
|
end;
|
|
exceptflowcontrol:=flowcontrol;
|
|
|
|
if fc_exit in exceptflowcontrol then
|
|
begin
|
|
{ do some magic for exit in the try block }
|
|
cg.a_label(current_asmdata.CurrAsmList,exitexceptlabel);
|
|
cg.g_call(current_asmdata.CurrAsmList,'FPC_DONEEXCEPTION');
|
|
if (fc_unwind_exit in oldflowcontrol) then
|
|
cg.g_local_unwind(current_asmdata.CurrAsmList,oldCurrExitLabel)
|
|
else
|
|
cg.a_jmp_always(current_asmdata.CurrAsmList,oldCurrExitLabel);
|
|
end;
|
|
|
|
if fc_break in exceptflowcontrol then
|
|
begin
|
|
cg.a_label(current_asmdata.CurrAsmList,breakexceptlabel);
|
|
cg.g_call(current_asmdata.CurrAsmList,'FPC_DONEEXCEPTION');
|
|
if (fc_unwind_loop in oldflowcontrol) then
|
|
cg.g_local_unwind(current_asmdata.CurrAsmList,oldBreakLabel)
|
|
else
|
|
cg.a_jmp_always(current_asmdata.CurrAsmList,oldBreakLabel);
|
|
end;
|
|
|
|
if fc_continue in exceptflowcontrol then
|
|
begin
|
|
cg.a_label(current_asmdata.CurrAsmList,continueexceptlabel);
|
|
cg.g_call(current_asmdata.CurrAsmList,'FPC_DONEEXCEPTION');
|
|
if (fc_unwind_loop in oldflowcontrol) then
|
|
cg.g_local_unwind(current_asmdata.CurrAsmList,oldContinueLabel)
|
|
else
|
|
cg.a_jmp_always(current_asmdata.CurrAsmList,oldContinueLabel);
|
|
end;
|
|
|
|
emit_nop;
|
|
cg.a_label(current_asmdata.CurrAsmList,endexceptlabel);
|
|
tcpuprocinfo(current_procinfo).add_except_scope(trylabel,exceptlabel,endexceptlabel,filterlabel);
|
|
|
|
errorexit:
|
|
{ restore all saved labels }
|
|
endexceptlabel:=oldendexceptlabel;
|
|
|
|
{ restore the control flow labels }
|
|
current_procinfo.CurrExitLabel:=oldCurrExitLabel;
|
|
if assigned(oldBreakLabel) then
|
|
begin
|
|
current_procinfo.CurrContinueLabel:=oldContinueLabel;
|
|
current_procinfo.CurrBreakLabel:=oldBreakLabel;
|
|
end;
|
|
|
|
{ return all used control flow statements }
|
|
flowcontrol:=oldflowcontrol+(exceptflowcontrol +
|
|
tryflowcontrol - [fc_inflowcontrol]);
|
|
end;
|
|
|
|
initialization
|
|
craisenode:=taarch64raisenode;
|
|
connode:=taarch64onnode;
|
|
ctryexceptnode:=taarch64tryexceptnode;
|
|
ctryfinallynode:=taarch64tryfinallynode;
|
|
end.
|
|
|