mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-10-24 05:37:41 +02:00

to "nil", so that returning them to Java code does not return a nil pointer git-svn-id: branches/jvmbackend@18417 -
523 lines
18 KiB
ObjectPascal
523 lines
18 KiB
ObjectPascal
{
|
|
Copyright (c) 1998-2011 by Florian Klaempfl and Jonas Maebe
|
|
|
|
Generate JVM inline nodes
|
|
|
|
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 njvminl;
|
|
|
|
{$i fpcdefs.inc}
|
|
|
|
interface
|
|
|
|
uses
|
|
cpubase,
|
|
node,ninl,ncginl;
|
|
|
|
type
|
|
tjvminlinenode = class(tcginlinenode)
|
|
protected
|
|
function typecheck_length(var handled: boolean): tnode;
|
|
function typecheck_high(var handled: boolean): tnode;
|
|
function typecheck_new(var handled: boolean): tnode;
|
|
|
|
function first_setlength_array: tnode;
|
|
public
|
|
{ typecheck override to intercept handling }
|
|
function pass_typecheck: tnode; override;
|
|
|
|
{ first pass override
|
|
so that the code generator will actually generate
|
|
these nodes.
|
|
}
|
|
(*
|
|
function first_sqrt_real: tnode; override;
|
|
*)
|
|
function first_sqr_real: tnode; override;
|
|
function first_trunc_real: tnode; override;
|
|
(*
|
|
function first_round_real: tnode; override;
|
|
*)
|
|
function first_new: tnode; override;
|
|
function first_setlength: tnode; override;
|
|
|
|
procedure second_length; override;
|
|
(*
|
|
procedure second_sqrt_real; override;
|
|
procedure second_abs_real; override;
|
|
*)
|
|
procedure second_sqr_real; override;
|
|
procedure second_trunc_real; override;
|
|
(*
|
|
procedure second_round_real; override;
|
|
*)
|
|
procedure second_new; override;
|
|
procedure second_setlength; override;
|
|
protected
|
|
procedure load_fpu_location;
|
|
end;
|
|
|
|
implementation
|
|
|
|
uses
|
|
cutils,globals,verbose,globtype,constexp,
|
|
aasmbase,aasmtai,aasmdata,aasmcpu,
|
|
symtype,symconst,symdef,symsym,symtable,jvmdef,
|
|
defutil,
|
|
nbas,ncon,ncnv,ncal,nld,nflw,nutils,
|
|
cgbase,pass_1,pass_2,
|
|
cpuinfo,ncgutil,
|
|
cgutils,hlcgobj,hlcgcpu;
|
|
|
|
|
|
{*****************************************************************************
|
|
tjvminlinenode
|
|
*****************************************************************************}
|
|
|
|
function tjvminlinenode.typecheck_length(var handled: boolean): tnode;
|
|
begin
|
|
typecheckpass(left);
|
|
if is_dynamic_array(left.resultdef) or
|
|
is_open_array(left.resultdef) then
|
|
begin
|
|
resultdef:=s32inttype;
|
|
result:=nil;
|
|
handled:=true;
|
|
end;
|
|
end;
|
|
|
|
|
|
function tjvminlinenode.typecheck_high(var handled: boolean): tnode;
|
|
begin
|
|
typecheckpass(left);
|
|
if is_dynamic_array(left.resultdef) or
|
|
is_open_array(left.resultdef) then
|
|
begin
|
|
{ replace with pred(length(arr)) }
|
|
result:=cinlinenode.create(in_pred_x,false,
|
|
cinlinenode.create(in_length_x,false,left));
|
|
left:=nil;
|
|
handled:=true;
|
|
end;
|
|
end;
|
|
|
|
|
|
function tjvminlinenode.typecheck_new(var handled: boolean): tnode;
|
|
var
|
|
para: tcallparanode;
|
|
elemdef: tdef;
|
|
begin
|
|
{ normally never exists; used by the JVM backend to create new
|
|
arrays because it requires special opcodes }
|
|
tcallparanode(left).get_paratype;
|
|
if is_dynamic_array(left.resultdef) then
|
|
begin
|
|
para:=tcallparanode(left);
|
|
{ need at least one extra parameter in addition to the
|
|
array }
|
|
if not assigned(para.right) then
|
|
internalerror(2011012206);
|
|
elemdef:=tarraydef(left.resultdef).elementdef;
|
|
while elemdef.typ=arraydef do
|
|
begin
|
|
{ if we have less length specifiers than dimensions, make
|
|
the last array an array of length 0 }
|
|
if not assigned(para.right) then
|
|
begin
|
|
para.right:=ccallparanode.create(
|
|
cordconstnode.create(0,s32inttype,false),nil);
|
|
tcallparanode(para.right).get_paratype;
|
|
break;
|
|
end
|
|
else
|
|
begin
|
|
inserttypeconv(tcallparanode(para.right).left,s32inttype);
|
|
tcallparanode(para.right).get_paratype;
|
|
end;
|
|
para:=tcallparanode(para.right);
|
|
elemdef:=tarraydef(elemdef).elementdef;
|
|
end;
|
|
result:=nil;
|
|
resultdef:=left.resultdef;
|
|
handled:=true;
|
|
end;
|
|
end;
|
|
|
|
|
|
function tjvminlinenode.pass_typecheck: tnode;
|
|
var
|
|
handled: boolean;
|
|
begin
|
|
handled:=false;
|
|
case inlinenumber of
|
|
in_length_x:
|
|
begin
|
|
result:=typecheck_length(handled);
|
|
end;
|
|
in_high_x:
|
|
begin
|
|
result:=typecheck_high(handled);
|
|
end;
|
|
in_new_x:
|
|
begin
|
|
result:=typecheck_new(handled);
|
|
end;
|
|
end;
|
|
if not handled then
|
|
result:=inherited pass_typecheck;
|
|
end;
|
|
|
|
|
|
(*
|
|
function tjvminlinenode.first_sqrt_real : tnode;
|
|
begin
|
|
if (current_settings.cputype >= cpu_PPC970) then
|
|
begin
|
|
expectloc:=LOC_FPUREGISTER;
|
|
first_sqrt_real := nil;
|
|
end
|
|
else
|
|
result:=inherited first_sqrt_real;
|
|
end;
|
|
*)
|
|
|
|
function tjvminlinenode.first_sqr_real : tnode;
|
|
begin
|
|
expectloc:=LOC_FPUREGISTER;
|
|
first_sqr_real:=nil;
|
|
end;
|
|
|
|
|
|
function tjvminlinenode.first_trunc_real : tnode;
|
|
begin
|
|
expectloc:=LOC_REGISTER;
|
|
first_trunc_real:=nil;
|
|
end;
|
|
|
|
|
|
function tjvminlinenode.first_new: tnode;
|
|
begin
|
|
{ skip the array; it's a type node }
|
|
tcallparanode(tcallparanode(left).right).firstcallparan;
|
|
expectloc:=LOC_REGISTER;
|
|
result:=nil;
|
|
end;
|
|
|
|
|
|
function tjvminlinenode.first_setlength_array: tnode;
|
|
var
|
|
assignmenttarget,
|
|
ppn,
|
|
newparas: tnode;
|
|
newnode: tnode;
|
|
eledef,
|
|
objarraydef: tdef;
|
|
ndims: longint;
|
|
finaltype: char;
|
|
setlenroutine: string;
|
|
lefttemp: ttempcreatenode;
|
|
newblock: tblocknode;
|
|
newstatement: tstatementnode;
|
|
primitive: boolean;
|
|
begin
|
|
{ first parameter is the array, the rest are the dimensions }
|
|
newparas:=tcallparanode(left).right;
|
|
tcallparanode(left).right:=nil;
|
|
{ count the number of specified dimensions, and determine the type of
|
|
the final one }
|
|
ppn:=newparas;
|
|
eledef:=tarraydef(left.resultdef).elementdef;
|
|
{ ppn already points to the first dimension }
|
|
ndims:=1;
|
|
while assigned(tcallparanode(ppn).right) do
|
|
begin
|
|
inc(ndims);
|
|
eledef:=tarraydef(eledef).elementdef;
|
|
ppn:=tcallparanode(ppn).right;
|
|
end;
|
|
{ in case it's a dynamic array of static arrays, we must also allocate
|
|
the static arrays! }
|
|
while (eledef.typ=arraydef) and
|
|
not is_dynamic_array(eledef) do
|
|
begin
|
|
inc(ndims);
|
|
tcallparanode(ppn).right:=
|
|
ccallparanode.create(
|
|
genintconstnode(tarraydef(eledef).elecount),nil);
|
|
ppn:=tcallparanode(ppn).right;
|
|
eledef:=tarraydef(eledef).elementdef;
|
|
end;
|
|
{ prepend type parameter for the array }
|
|
newparas:=ccallparanode.create(ctypenode.create(left.resultdef),newparas);
|
|
ttypenode(tcallparanode(newparas).left).allowed:=true;
|
|
{ node to create the new array }
|
|
newnode:=cinlinenode.create(in_new_x,false,newparas);
|
|
{ Common parameters for setlength helper }
|
|
{ start with org (save assignmenttarget itself to assign the result back to) }
|
|
{ store left into a temp in case it may contain a function call
|
|
(which must not be evaluated twice) }
|
|
lefttemp:=maybereplacewithtempref(tcallparanode(left).left,tcallparanode(left).left.resultdef.size,false);
|
|
if assigned(lefttemp) then
|
|
begin
|
|
newblock:=internalstatements(newstatement);
|
|
addstatement(newstatement,lefttemp);
|
|
assignmenttarget:=ctemprefnode.create(lefttemp);
|
|
typecheckpass(tnode(assignmenttarget));
|
|
end
|
|
else
|
|
assignmenttarget:=tcallparanode(left).left.getcopy;
|
|
newparas:=left;
|
|
left:=nil;
|
|
{ if more than 1 dimension, or if 1 dimention of a non-primitive type,
|
|
typecast to generic array of tobject }
|
|
setlenroutine:=jvmarrtype(eledef,primitive);
|
|
if (ndims>1) or
|
|
not primitive then
|
|
begin
|
|
objarraydef:=search_system_type('TJOBJECTARRAY').typedef;
|
|
tcallparanode(newparas).left:=ctypeconvnode.create_explicit(tcallparanode(newparas).left,objarraydef);
|
|
newnode:=ctypeconvnode.create_explicit(newnode,objarraydef);
|
|
end;
|
|
{ prepend new }
|
|
newparas:=ccallparanode.create(newnode,newparas);
|
|
{ prepend deepcopy }
|
|
newparas:=ccallparanode.create(cordconstnode.create(0,pasbool8type,false),newparas);
|
|
{ call the right setlenght helper }
|
|
if ndims>1 then
|
|
begin
|
|
finaltype:=jvmarrtype_setlength(eledef);
|
|
setlenroutine:='FPC_SETLENGTH_DYNARR_MULTIDIM';
|
|
{ create proper parameters, from right to left:
|
|
eletype=finaltype, ndim=ndims, deepcopy=false, new=newnode,
|
|
assignmenttarget=tcallparanode(left).left }
|
|
{ prepend ndim }
|
|
newparas:=ccallparanode.create(cordconstnode.create(ndims,s32inttype,false),newparas);
|
|
{ prepend eletype }
|
|
newparas:=ccallparanode.create(cordconstnode.create(ord(finaltype),cwidechartype,false),newparas);
|
|
end
|
|
else
|
|
begin
|
|
if not primitive then
|
|
setlenroutine:='OBJECT'
|
|
else
|
|
uppervar(setlenroutine);
|
|
setlenroutine:='FPC_SETLENGTH_DYNARR_J'+setlenroutine;
|
|
{ create proper parameters, from right to left:
|
|
deepcopy=false, new=newnode, assignmenttarget=tcallparnode(left).left
|
|
-> already done in common part above }
|
|
end;
|
|
result:=ccallnode.createintern(setlenroutine,newparas);
|
|
{ assign result back to org (no call-by-reference for Java) }
|
|
result:=cassignmentnode.create(assignmenttarget,
|
|
ctypeconvnode.create_explicit(result,assignmenttarget.resultdef));
|
|
if assigned(lefttemp) then
|
|
begin
|
|
addstatement(newstatement,result);
|
|
addstatement(newstatement,ctempdeletenode.create(lefttemp));
|
|
result:=newblock;
|
|
end;
|
|
end;
|
|
|
|
|
|
function tjvminlinenode.first_setlength: tnode;
|
|
|
|
begin
|
|
{ reverse the parameter order so we can process them more easily }
|
|
left:=reverseparameters(tcallparanode(left));
|
|
{ treat setlength(x,0) specially: used to init uninitialised locations }
|
|
if not assigned(tcallparanode(tcallparanode(left).right).right) and
|
|
is_constintnode(tcallparanode(tcallparanode(left).right).left) and
|
|
(tordconstnode(tcallparanode(tcallparanode(left).right).left).value=0) then
|
|
begin
|
|
result:=nil;
|
|
expectloc:=LOC_VOID;
|
|
exit;
|
|
end;
|
|
case left.resultdef.typ of
|
|
arraydef:
|
|
result:=first_setlength_array;
|
|
else
|
|
internalerror(2011031204);
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure tjvminlinenode.second_length;
|
|
begin
|
|
if is_dynamic_array(left.resultdef) or
|
|
is_open_array(left.resultdef) then
|
|
begin
|
|
location_reset(location,LOC_REGISTER,OS_S32);
|
|
location.register:=hlcg.getintregister(current_asmdata.CurrAsmList,s32inttype);
|
|
secondpass(left);
|
|
thlcgjvm(hlcg).g_getarraylen(current_asmdata.CurrAsmList,left.location);
|
|
thlcgjvm(hlcg).a_load_stack_reg(current_asmdata.CurrAsmList,resultdef,location.register);
|
|
end
|
|
else
|
|
internalerror(2011012004);
|
|
end;
|
|
|
|
(*
|
|
function tjvminlinenode.first_round_real : tnode;
|
|
begin
|
|
if (current_settings.cputype >= cpu_PPC970) then
|
|
begin
|
|
expectloc:=LOC_REFERENCE;
|
|
first_round_real := nil;
|
|
end
|
|
else
|
|
result:=inherited first_round_real;
|
|
end;
|
|
*)
|
|
|
|
{ load the FPU value on the evaluation stack }
|
|
procedure tjvminlinenode.load_fpu_location;
|
|
begin
|
|
secondpass(left);
|
|
thlcgjvm(hlcg).a_load_loc_stack(current_asmdata.CurrAsmList,left.resultdef,left.location);
|
|
end;
|
|
|
|
(*
|
|
procedure tjvminlinenode.second_sqrt_real;
|
|
begin
|
|
if (current_settings.cputype < cpu_PPC970) then
|
|
internalerror(2007020910);
|
|
location.loc:=LOC_FPUREGISTER;
|
|
load_fpu_location;
|
|
case left.location.size of
|
|
OS_F32:
|
|
current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_FSQRTS,location.register,
|
|
left.location.register));
|
|
OS_F64:
|
|
current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_FSQRT,location.register,
|
|
left.location.register));
|
|
else
|
|
inherited;
|
|
end;
|
|
end;
|
|
*)
|
|
|
|
procedure tjvminlinenode.second_sqr_real;
|
|
begin
|
|
load_fpu_location;
|
|
location_reset(location,LOC_FPUREGISTER,location.size);
|
|
location.register:=hlcg.getfpuregister(current_asmdata.CurrAsmList,resultdef);
|
|
case left.location.size of
|
|
OS_F32:
|
|
begin
|
|
current_asmdata.CurrAsmList.concat(taicpu.op_none(a_dup));
|
|
thlcgjvm(hlcg).incstack(current_asmdata.CurrAsmList,1);
|
|
current_asmdata.CurrAsmList.concat(taicpu.op_none(a_fmul));
|
|
thlcgjvm(hlcg).decstack(current_asmdata.CurrAsmList,1);
|
|
end;
|
|
OS_F64:
|
|
begin
|
|
current_asmdata.CurrAsmList.concat(taicpu.op_none(a_dup2));
|
|
thlcgjvm(hlcg).incstack(current_asmdata.CurrAsmList,2);
|
|
current_asmdata.CurrAsmList.concat(taicpu.op_none(a_dmul));
|
|
thlcgjvm(hlcg).decstack(current_asmdata.CurrAsmList,2);
|
|
end;
|
|
else
|
|
internalerror(2011010804);
|
|
end;
|
|
thlcgjvm(hlcg).a_load_stack_reg(current_asmdata.CurrAsmList,resultdef,location.register);
|
|
end;
|
|
|
|
|
|
procedure tjvminlinenode.second_trunc_real;
|
|
begin
|
|
load_fpu_location;
|
|
location_reset(location,LOC_REGISTER,left.location.size);
|
|
location.register:=hlcg.getintregister(current_asmdata.CurrAsmList,resultdef);
|
|
case left.location.size of
|
|
OS_F32:
|
|
begin
|
|
current_asmdata.CurrAsmList.concat(taicpu.op_none(a_f2l));
|
|
{ 32 bit float -> 64 bit int: +1 stack slot }
|
|
thlcgjvm(hlcg).incstack(current_asmdata.CurrAsmList,1);
|
|
end;
|
|
OS_F64:
|
|
begin
|
|
{ 64 bit float -> 64 bit int: same number of stack slots }
|
|
current_asmdata.CurrAsmList.concat(taicpu.op_none(a_d2l));
|
|
end;
|
|
else
|
|
internalerror(2011010805);
|
|
end;
|
|
thlcgjvm(hlcg).a_load_stack_reg(current_asmdata.CurrAsmList,resultdef,location.register);
|
|
end;
|
|
|
|
|
|
procedure tjvminlinenode.second_new;
|
|
var
|
|
arr: tnode;
|
|
hp: tcallparanode;
|
|
paracount: longint;
|
|
begin
|
|
hp:=tcallparanode(left);
|
|
{ we don't second pass this one, it's only a type node }
|
|
arr:=hp.left;
|
|
if not is_dynamic_array(arr.resultdef) then
|
|
internalerror(2011012204);
|
|
hp:=tcallparanode(hp.right);
|
|
if not assigned(hp) then
|
|
internalerror(2011012205);
|
|
paracount:=0;
|
|
{ put all the dimensions on the stack }
|
|
repeat
|
|
inc(paracount);
|
|
secondpass(hp.left);
|
|
thlcgjvm(hlcg).a_load_loc_stack(current_asmdata.CurrAsmList,hp.left.resultdef,hp.left.location);
|
|
hp:=tcallparanode(hp.right);
|
|
until not assigned(hp);
|
|
{ create the array }
|
|
thlcgjvm(hlcg).g_newarray(current_asmdata.CurrAsmList,arr.resultdef,paracount);
|
|
location_reset(location,LOC_REGISTER,OS_ADDR);
|
|
location.register:=hlcg.getaddressregister(current_asmdata.CurrAsmList,resultdef);
|
|
thlcgjvm(hlcg).a_load_stack_reg(current_asmdata.CurrAsmList,arr.resultdef,location.register);
|
|
end;
|
|
|
|
|
|
procedure tjvminlinenode.second_setlength;
|
|
var
|
|
target: tnode;
|
|
lenpara: tnode;
|
|
begin
|
|
target:=tcallparanode(left).left;
|
|
lenpara:=tcallparanode(tcallparanode(left).right).left;
|
|
if assigned(tcallparanode(tcallparanode(left).right).right) or
|
|
not is_constintnode(lenpara) or
|
|
(tordconstnode(lenpara).value<>0) then
|
|
internalerror(2011031801);
|
|
|
|
secondpass(target);
|
|
if is_dynamic_array(target.resultdef) then
|
|
begin
|
|
thlcgjvm(hlcg).a_load_const_stack(current_asmdata.CurrAsmList,s32inttype,0,R_INTREGISTER);
|
|
thlcgjvm(hlcg).g_newarray(current_asmdata.CurrAsmList,target.resultdef,1);
|
|
end
|
|
else
|
|
internalerror(2011031401);
|
|
thlcgjvm(hlcg).a_load_stack_loc(current_asmdata.CurrAsmList,target.resultdef,target.location);
|
|
end;
|
|
|
|
|
|
begin
|
|
cinlinenode:=tjvminlinenode;
|
|
end.
|