fpc/compiler/xtensa/cpupara.pas
2020-08-30 20:53:59 +00:00

548 lines
19 KiB
ObjectPascal

{
Copyright (c) 2002 by Florian Klaempfl
Xtensa specific calling conventions
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 cpupara;
{$i fpcdefs.inc}
interface
uses
globtype,
aasmtai,aasmdata,
cpubase,
symconst,symtype,symdef,symsym,
paramgr,parabase,cgbase,cgutils;
type
tcpuparamanager = class(tparamanager)
function get_volatile_registers_int(calloption : tproccalloption):tcpuregisterset;override;
function get_volatile_registers_fpu(calloption : tproccalloption):tcpuregisterset;override;
function push_addr_param(varspez:tvarspez;def : tdef;calloption : tproccalloption) : boolean;override;
function create_paraloc_info(p : tabstractprocdef; side: tcallercallee):longint;override;
function create_varargs_paraloc_info(p : tabstractprocdef; side: tcallercallee; varargspara:tvarargsparalist):longint;override;
function get_funcretloc(p : tabstractprocdef; side: tcallercallee; forcetempdef: tdef): tcgpara;override;
function ret_in_param(def: tdef; pd: tabstractprocdef): boolean;override;
private
{ the max. register depends on the used call instruction }
maxintreg : TSuperRegister;
procedure init_values(p: tabstractprocdef; side: tcallercallee; var curintreg: tsuperregister; var cur_stack_offset: aword);
function create_paraloc_info_intern(p : tabstractprocdef; side : tcallercallee;
paras : tparalist; var curintreg : tsuperregister;
var cur_stack_offset : aword; varargsparas : boolean) : longint;
end;
implementation
uses
cpuinfo,globals,
verbose,systems,
defutil,
symtable,symcpu,
procinfo,cpupi;
function tcpuparamanager.get_volatile_registers_int(calloption : tproccalloption):tcpuregisterset;
begin
{ we have actually to check what calling instruction is used, but we do not handle this,
instead CALL(X)8 is used always }
if target_info.abi=abi_xtensa_windowed then
result:=[RS_A8..RS_A15]
else
result:=[RS_A0..RS_A11];
end;
function tcpuparamanager.get_volatile_registers_fpu(calloption : tproccalloption):tcpuregisterset;
begin
result:=[RS_F0..RS_F15];
end;
function getparaloc(p : tdef) : tcgloc;
begin
{ Later, the LOC_REFERENCE is in most cases changed into LOC_REGISTER
if push_addr_param for the def is true
}
case p.typ of
orddef:
result:=LOC_REGISTER;
floatdef:
result:=LOC_REGISTER;
enumdef:
result:=LOC_REGISTER;
pointerdef:
result:=LOC_REGISTER;
formaldef:
result:=LOC_REGISTER;
classrefdef:
result:=LOC_REGISTER;
procvardef:
result:=LOC_REGISTER;
recorddef:
if p.size>24 then
result:=LOC_REFERENCE
else
result:=LOC_REGISTER;
objectdef:
if is_object(p) and (p.size>24) then
result:=LOC_REFERENCE
else
result:=LOC_REGISTER;
stringdef:
if is_shortstring(p) or is_longstring(p) then
result:=LOC_REFERENCE
else
result:=LOC_REGISTER;
filedef:
result:=LOC_REGISTER;
arraydef:
if is_dynamic_array(p) or (p.size<=24) then
getparaloc:=LOC_REGISTER
else
result:=LOC_REFERENCE;
setdef:
if is_smallset(p) then
result:=LOC_REGISTER
else
result:=LOC_REFERENCE;
variantdef:
result:=LOC_REGISTER;
{ avoid problems with errornous definitions }
errordef:
result:=LOC_REGISTER;
else
internalerror(2020082501);
end;
end;
function tcpuparamanager.push_addr_param(varspez:tvarspez;def : tdef;calloption : tproccalloption) : boolean;
begin
result:=false;
{ var,out,constref always require address }
if varspez in [vs_var,vs_out,vs_constref] then
begin
result:=true;
exit;
end;
case def.typ of
formaldef :
result:=true;
recorddef :
result:=(varspez = vs_const);
arraydef:
result:=((varspez = vs_const) and (tarraydef(def).highrange>=tarraydef(def).lowrange)) or
is_open_array(def) or
is_array_of_const(def) or
is_array_constructor(def);
objectdef :
result:=is_object(def) and (varspez = vs_const);
variantdef,
setdef :
result:=(varspez = vs_const);
stringdef :
result:=tstringdef(def).stringtype in [st_shortstring,st_longstring];
else
;
end;
end;
procedure tcpuparamanager.init_values(p : tabstractprocdef; side : tcallercallee; var curintreg: tsuperregister; var cur_stack_offset: aword);
begin
cur_stack_offset:=0;
case target_info.abi of
abi_xtensa_windowed:
begin
if side=calleeside then
begin
curintreg:=RS_A2;
maxintreg:=RS_A7;
if current_procinfo.framepointer=NR_STACK_POINTER_REG then
cur_stack_offset:=(p as tcpuprocdef).total_stackframe_size;
end
else
begin
{ we use CALL(X)8 only so far }
curintreg:=RS_A10;
maxintreg:=RS_A15;
end;
end;
abi_xtensa_call0:
begin
curintreg:=RS_A2;
maxintreg:=RS_A7;
end;
else
Internalerror(2020031404);
end;
end;
function tcpuparamanager.get_funcretloc(p : tabstractprocdef; side: tcallercallee; forcetempdef: tdef): tcgpara;
var
paraloc : pcgparalocation;
retcgsize : tcgsize;
begin
if set_common_funcretloc_info(p,forcetempdef,retcgsize,result) then
exit;
paraloc:=result.add_location;
if retcgsize in [OS_64,OS_S64,OS_F64] then
begin
{ low 32bits }
paraloc^.loc:=LOC_REGISTER;
paraloc^.size:=OS_32;
paraloc^.def:=u32inttype;
if side=callerside then
case target_info.abi of
abi_xtensa_call0:
paraloc^.register:=NR_A2;
abi_xtensa_windowed:
{ only call8 used/supported so far }
paraloc^.register:=newreg(R_INTREGISTER,RS_A10,cgsize2subreg(R_INTREGISTER,retcgsize));
else
Internalerror(2020032201);
end
else
paraloc^.register:=NR_A2;
{ high 32bits }
paraloc:=result.add_location;
paraloc^.loc:=LOC_REGISTER;
paraloc^.size:=OS_32;
paraloc^.def:=u32inttype;
if side=callerside then
case target_info.abi of
abi_xtensa_call0:
paraloc^.register:=NR_A3;
abi_xtensa_windowed:
{ only call8 used/supported so far }
paraloc^.register:=newreg(R_INTREGISTER,RS_A11,cgsize2subreg(R_INTREGISTER,retcgsize));
else
Internalerror(2020032202);
end
else
paraloc^.register:=NR_A3;
end
else
begin
paraloc^.loc:=LOC_REGISTER;
if side=callerside then
case target_info.abi of
abi_xtensa_call0:
paraloc^.register:=newreg(R_INTREGISTER,RS_FUNCTION_RESULT_REG,cgsize2subreg(R_INTREGISTER,retcgsize));
abi_xtensa_windowed:
{ only call8 used/supported so far }
paraloc^.register:=newreg(R_INTREGISTER,RS_A10,cgsize2subreg(R_INTREGISTER,retcgsize));
else
Internalerror(2020031502);
end
else
paraloc^.register:=newreg(R_INTREGISTER,RS_FUNCTION_RETURN_REG,cgsize2subreg(R_INTREGISTER,retcgsize));
paraloc^.size:=OS_32;
paraloc^.def:=result.def;
end;
end;
function tcpuparamanager.ret_in_param(def:tdef;pd:tabstractprocdef):boolean;
var
i: longint;
sym: tsym;
basedef: tdef;
begin
if handle_common_ret_in_param(def,pd,result) then
exit;
case def.typ of
arraydef,
objectdef,
stringdef,
setdef,
recorddef:
result:=def.size>16;
floatdef,
variantdef,
procvardef:
result:=false
else
result:=inherited ret_in_param(def,pd);
end;
end;
function tcpuparamanager.create_paraloc_info(p : tabstractprocdef; side: tcallercallee):longint;
var
cur_stack_offset: aword;
curintreg: tsuperregister;
begin
init_values(p,side,curintreg,cur_stack_offset);
result:=create_paraloc_info_intern(p,side,p.paras,curintreg,cur_stack_offset,false);
if result<cur_stack_offset then
Internalerror(2020083001);
create_funcretloc_info(p,side);
end;
function tcpuparamanager.create_paraloc_info_intern(p : tabstractprocdef; side: tcallercallee; paras:tparalist;
var curintreg: tsuperregister; var cur_stack_offset: aword; varargsparas: boolean):longint;
var
stack_offset: longint;
paralen: aint;
nextintreg : tsuperregister;
locdef,
fdef,
paradef : tdef;
paraloc : pcgparalocation;
i : integer;
hp : tparavarsym;
loc : tcgloc;
paracgsize: tcgsize;
firstparaloc: boolean;
begin
{$ifdef extdebug}
if po_explicitparaloc in p.procoptions then
internalerror(200411141);
{$endif extdebug}
loc.reset;
{ currently only support C-style array of const }
if (p.proccalloption in cstylearrayofconst) and
is_array_of_const(paradef) then
begin
paraloc:=loc.add_location;
{ hack: the paraloc must be valid, but is not actually used }
paraloc^.loc := LOC_REGISTER;
paraloc^.register := NR_A2;
paraloc^.size := OS_ADDR;
paraloc^.def:=voidpointertype;
result:=cur_stack_offset;
exit;
end;
result:=0;
nextintreg := curintreg;
stack_offset := cur_stack_offset;
for i:=0 to paras.count-1 do
begin
hp:=tparavarsym(paras[i]);
paradef := hp.vardef;
hp.paraloc[side].reset;
{ currently only support C-style array of const }
if (p.proccalloption in cstylearrayofconst) and
is_array_of_const(paradef) then
begin
paraloc:=hp.paraloc[side].add_location;
{ hack: the paraloc must be valid, but is not actually used }
paraloc^.loc := LOC_REGISTER;
paraloc^.register := NR_A2;
paraloc^.size := OS_ADDR;
paraloc^.def:=voidpointertype;
break;
end;
if push_addr_param(hp.varspez,paradef,p.proccalloption) then
begin
paradef:=cpointerdef.getreusable_no_free(paradef);
loc:=LOC_REGISTER;
paracgsize := OS_ADDR;
paralen := tcgsize2size[OS_ADDR];
end
else
begin
if not is_special_array(paradef) then
paralen := paradef.size
else
paralen := tcgsize2size[def_cgsize(paradef)];
if (paradef.typ in [objectdef,arraydef,recorddef,setdef,stringdef]) and
not is_special_array(paradef) and
(hp.varspez in [vs_value,vs_const]) then
paracgsize:=int_cgsize(paralen)
else
begin
paracgsize:=def_cgsize(paradef);
if (paracgsize=OS_NO) then
begin
paracgsize:=OS_ADDR;
paralen := tcgsize2size[OS_ADDR];
paradef:=voidpointertype;
end;
end;
end;
loc:=getparaloc(paradef);
if (loc=LOC_REGISTER) and ((maxintreg-nextintreg+1)*4<paradef.size) then
begin
loc:=LOC_REFERENCE;
nextintreg:=maxintreg+1;
end;
hp.paraloc[side].alignment:=std_param_align;
hp.paraloc[side].size:=paracgsize;
hp.paraloc[side].intsize:=paralen;
hp.paraloc[side].def:=paradef;
if (loc=LOC_REGISTER) and (is_64bit(paradef)) and
odd(nextintreg-RS_A2) then
inc(nextintreg);
if (paralen = 0) then
if (paradef.typ = recorddef) then
begin
paraloc:=hp.paraloc[side].add_location;
paraloc^.loc := LOC_VOID;
end
else
internalerror(2020031407);
locdef:=paradef;
firstparaloc:=true;
{ can become < 0 for e.g. 3-byte records }
while (paralen > 0) do
begin
paraloc:=hp.paraloc[side].add_location;
{ In case of po_delphi_nested_cc, the parent frame pointer
is always passed on the stack. }
if (loc = LOC_REGISTER) and
(nextintreg <= maxintreg) and
(not(vo_is_parentfp in hp.varoptions) or
not(po_delphi_nested_cc in p.procoptions)) then
begin
paraloc^.loc := loc;
{ make sure we don't lose whether or not the type is signed }
if (paradef.typ<>orddef) then
begin
paracgsize:=int_cgsize(paralen);
locdef:=get_paraloc_def(paradef,paralen,firstparaloc);
end;
if (paracgsize in [OS_NO,OS_64,OS_S64,OS_128,OS_S128]) then
begin
paraloc^.size:=OS_INT;
paraloc^.def:=u32inttype;
end
else
begin
paraloc^.size:=paracgsize;
paraloc^.def:=locdef;
end;
paraloc^.register:=newreg(R_INTREGISTER,nextintreg,R_SUBNONE);
inc(nextintreg);
dec(paralen,tcgsize2size[paraloc^.size]);
end
else { LOC_REFERENCE }
begin
paraloc^.loc:=LOC_REFERENCE;
case loc of
LOC_REGISTER,
LOC_REFERENCE:
begin
paraloc^.size:=int_cgsize(paralen);
if paraloc^.size<>OS_NO then
paraloc^.def:=cgsize_orddef(paraloc^.size)
else
paraloc^.def:=carraydef.getreusable_no_free(u8inttype,paralen);
end;
else
internalerror(2020031405);
end;
if side = callerside then
paraloc^.reference.index:=NR_STACK_POINTER_REG
else
paraloc^.reference.index:=current_procinfo.framepointer;
paraloc^.reference.offset:=stack_offset;
inc(stack_offset,align(paralen,4));
while (paralen > 0) and
(nextintreg < maxintreg) do
begin
inc(nextintreg);
dec(paralen,sizeof(pint));
end;
paralen := 0;
end;
<<<<<<< HEAD
firstparaloc:=false;
end;
end;
curintreg:=nextintreg;
cur_stack_offset:=stack_offset;
result:=stack_offset;
=======
paralen := 0;
end;
firstparaloc:=false;
end;
result:=cur_stack_offset;
end;
function tcpuparamanager.create_paraloc_info_intern(p : tabstractprocdef; side: tcallercallee; paras:tparalist;
var curintreg: tsuperregister; var cur_stack_offset: aword; varargsparas: boolean):longint;
var
i : integer;
begin
result:=0;
for i:=0 to paras.count-1 do
result:=create_paraloc1_info_intern(p,side,tparavarsym(paras[i]).vardef,tparavarsym(paras[i]).paraloc[side],tparavarsym(paras[i]).varspez,
tparavarsym(paras[i]).varoptions,curintreg,cur_stack_offset,false);
>>>>>>> 7962bf60e6... * squash stack frame size
end;
function tcpuparamanager.create_varargs_paraloc_info(p : tabstractprocdef; side: tcallercallee; varargspara:tvarargsparalist):longint;
var
cur_stack_offset: aword;
parasize, l: longint;
curintreg: tsuperregister;
i : integer;
hp: tparavarsym;
paraloc: pcgparalocation;
begin
init_values(p,side,curintreg,cur_stack_offset);
result:=create_paraloc_info_intern(p,side,p.paras,curintreg,cur_stack_offset, false);
if (p.proccalloption in cstylearrayofconst) then
{ just continue loading the parameters in the registers }
begin
if assigned(varargspara) then
begin
if side=callerside then
result:=create_paraloc_info_intern(p,side,varargspara,curintreg,cur_stack_offset,true)
else
internalerror(2020030704);
end;
end
else
internalerror(2020030703);
create_funcretloc_info(p,side);
end;
begin
paramanager:=tcpuparamanager.create;
end.