mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-06 07:47:59 +02:00

o support for ansistring constants. It's done via a detour because the JVM only supports UTF-16 string constants (no array of byte or anything like that): store every ansicharacter in the lower 8 bits of an UTF-16 constant string, and at run time copy the characters to an ansistring. The alternative is to generate code that stores every character separately to an array. o the base ansistring support is implemented in a class called AnsistringClass, and an ansistring is simply an instance of this class under the hood o the compiler currently does generate nil pointers as empty ansistrings unlike for unicodestrings, where we always explicitly generate an empty string. The reason is that unicodestrings are the same as JLString and hence common for Java interoperation, while ansistrings are unlikely to be used in interaction with external Java code * fixed indentation git-svn-id: branches/jvmbackend@18562 -
773 lines
27 KiB
ObjectPascal
773 lines
27 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_copy: tnode; override;
|
|
|
|
function handle_box: tnode; override;
|
|
function first_box: tnode; override;
|
|
|
|
function first_setlength_array: tnode;
|
|
function first_setlength_string: 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;
|
|
function first_length: 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;
|
|
procedure second_box; 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) or
|
|
is_wide_or_unicode_string(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.first_copy: tnode;
|
|
var
|
|
ppn: tcallparanode;
|
|
arr, len, start, kind: tnode;
|
|
eledef: tdef;
|
|
counter, ndims: longint;
|
|
finaltype: char;
|
|
begin
|
|
if is_dynamic_array(resultdef) then
|
|
begin
|
|
ppn:=tcallparanode(left);
|
|
counter:=1;
|
|
while assigned(ppn.right) do
|
|
begin
|
|
inc(counter);
|
|
ppn:=tcallparanode(ppn.right);
|
|
end;
|
|
if (counter=3) then
|
|
begin
|
|
len:=tcallparanode(left).left;
|
|
tcallparanode(left).left:=nil;
|
|
start:=tcallparanode(tcallparanode(left).right).left;
|
|
tcallparanode(tcallparanode(left).right).left:=nil;
|
|
{ free the original start/len paras and remove them }
|
|
ppn:=tcallparanode(left);
|
|
left:=tcallparanode(tcallparanode(left).right).right;
|
|
tcallparanode(ppn.right).right:=nil;
|
|
ppn.free;
|
|
end
|
|
else
|
|
begin
|
|
{ use special -1,-1 argument to copy the whole array }
|
|
len:=genintconstnode(-1);
|
|
start:=genintconstnode(-1);
|
|
end;
|
|
{ currently there is one parameter left: the array itself }
|
|
arr:=tcallparanode(left).left;
|
|
tcallparanode(left).left:=nil;
|
|
{ in case it's a dynamic array of static arrays, get the dimensions
|
|
of the static array components }
|
|
eledef:=tarraydef(resultdef).elementdef;
|
|
ndims:=1;
|
|
while (eledef.typ=arraydef) and
|
|
not is_dynamic_array(eledef) do
|
|
begin
|
|
inc(ndims);
|
|
eledef:=tarraydef(eledef).elementdef;
|
|
end;
|
|
{ get the final element kind }
|
|
finaltype:=jvmarrtype_setlength(eledef);
|
|
{ construct the call to
|
|
fpc_dynarray_copy(src: JLObject; start, len: longint; ndim: longint; eletype: jchar) }
|
|
result:=ccallnode.createintern('FPC_DYNARRAY_COPY',
|
|
ccallparanode.create(cordconstnode.create(ord(finaltype),cwidechartype,false),
|
|
ccallparanode.create(genintconstnode(ndims),
|
|
ccallparanode.create(len,
|
|
ccallparanode.create(start,
|
|
ccallparanode.create(ctypeconvnode.create_explicit(arr,java_jlobject),nil)
|
|
)
|
|
)
|
|
)
|
|
)
|
|
);
|
|
inserttypeconv_explicit(result,resultdef);
|
|
end
|
|
else
|
|
result:=inherited first_copy;
|
|
end;
|
|
|
|
|
|
function tjvminlinenode.handle_box: tnode;
|
|
begin
|
|
Result:=inherited;
|
|
resultdef:=java_jlobject;
|
|
end;
|
|
|
|
|
|
function tjvminlinenode.first_box: tnode;
|
|
begin
|
|
result:=nil;
|
|
expectloc:=LOC_REGISTER;
|
|
{$ifdef nounsupported}
|
|
internalerror(2011042603);
|
|
{$endif}
|
|
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;
|
|
finaltype:=jvmarrtype_setlength(eledef);
|
|
{ since the setlength prototypes require certain types, insert
|
|
explicit type conversions where necessary }
|
|
objarraydef:=nil;
|
|
if (ndims>1) then
|
|
begin
|
|
{ expects array of JLObject }
|
|
setlenroutine:='FPC_SETLENGTH_DYNARR_MULTIDIM';
|
|
objarraydef:=search_system_type('TJOBJECTARRAY').typedef
|
|
end
|
|
else
|
|
begin
|
|
if finaltype<>'R' then
|
|
begin
|
|
{ expects JLObject }
|
|
setlenroutine:='FPC_SETLENGTH_DYNARR_GENERIC';
|
|
objarraydef:=java_jlobject;
|
|
end
|
|
else
|
|
begin
|
|
{ expects array of FpcBaseRecord}
|
|
setlenroutine:='FPC_SETLENGTH_DYNARR_JRECORD';
|
|
objarraydef:=search_system_type('TJRECORDARRAY').typedef;
|
|
end;
|
|
end;
|
|
tcallparanode(newparas).left:=ctypeconvnode.create_explicit(tcallparanode(newparas).left,objarraydef);
|
|
newnode:=ctypeconvnode.create_explicit(newnode,objarraydef);
|
|
{ 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
|
|
{ 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
|
|
{ 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_string: tnode;
|
|
var
|
|
newblock: tblocknode;
|
|
newstatement: tstatementnode;
|
|
lefttemp: ttempcreatenode;
|
|
assignmenttarget: tnode;
|
|
stringtype: tstringdef;
|
|
begin
|
|
if is_wide_or_unicode_string(left.resultdef) or
|
|
is_ansistring(left.resultdef) then
|
|
begin
|
|
stringtype:=tstringdef(left.resultdef);
|
|
{ store left into a temp in case it may contain a function call
|
|
(which must not be evaluated twice) }
|
|
lefttemp:=maybereplacewithtempref(tcallparanode(left).left,stringtype.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;
|
|
{ back to original order for the call }
|
|
left:=reverseparameters(tcallparanode(left));
|
|
result:=cassignmentnode.create(assignmenttarget,
|
|
ccallnode.createintern('fpc_'+stringtype.stringtypname+'_setlength',left));
|
|
if assigned(lefttemp) then
|
|
begin
|
|
addstatement(newstatement,result);
|
|
addstatement(newstatement,ctempdeletenode.create(lefttemp));
|
|
result:=newblock;
|
|
end;
|
|
left:=nil;
|
|
end
|
|
{$ifndef nounsupported}
|
|
else if left.resultdef.typ=stringdef then
|
|
begin
|
|
result:=cnothingnode.create;
|
|
end
|
|
{$endif}
|
|
else
|
|
internalerror(2011031405);
|
|
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;
|
|
stringdef:
|
|
result:=first_setlength_string;
|
|
else
|
|
internalerror(2011031204);
|
|
end;
|
|
end;
|
|
|
|
|
|
function tjvminlinenode.first_length: tnode;
|
|
var
|
|
newblock: tblocknode;
|
|
newstatement: tstatementnode;
|
|
lentemp: ttempcreatenode;
|
|
ifcond,
|
|
stringnonnull,
|
|
stringnull: tnode;
|
|
psym: tsym;
|
|
stringclass: tdef;
|
|
begin
|
|
if is_wide_or_unicode_string(left.resultdef) or
|
|
is_ansistring(left.resultdef) then
|
|
begin
|
|
{ if assigned(stringclass(left)) then
|
|
lentemp:=stringclass(left).length()
|
|
else
|
|
lentemp:=0;
|
|
--> return lentemp
|
|
}
|
|
if is_ansistring(left.resultdef) then
|
|
stringclass:=java_ansistring
|
|
else
|
|
stringclass:=java_jlstring;
|
|
newblock:=internalstatements(newstatement);
|
|
lentemp:=ctempcreatenode.create(s32inttype,s32inttype.size,tt_persistent,true);
|
|
addstatement(newstatement,lentemp);
|
|
{ if-condition }
|
|
ifcond:=cinlinenode.create(in_assigned_x,false,
|
|
ccallparanode.create(ctypeconvnode.create_explicit(left.getcopy,stringclass),nil));
|
|
{ then-path (reuse left, since last use) }
|
|
psym:=search_struct_member(tabstractrecorddef(stringclass),'LENGTH');
|
|
if not assigned(psym) or
|
|
(psym.typ<>procsym) then
|
|
internalerror(2011031403);
|
|
stringnonnull:=cassignmentnode.create(
|
|
ctemprefnode.create(lentemp),
|
|
ccallnode.create(nil,tprocsym(psym),psym.owner,
|
|
ctypeconvnode.create_explicit(left,stringclass),[]));
|
|
left:=nil;
|
|
{ else-path}
|
|
stringnull:=cassignmentnode.create(
|
|
ctemprefnode.create(lentemp),
|
|
genintconstnode(0));
|
|
{ complete if-statement }
|
|
addstatement(newstatement,cifnode.create(ifcond,stringnonnull,stringnull));
|
|
{ return temp }
|
|
addstatement(newstatement,ctempdeletenode.create_normal_temp(lentemp));
|
|
addstatement(newstatement,ctemprefnode.create(lentemp));
|
|
result:=newblock;
|
|
end
|
|
{$ifndef nounsupported}
|
|
else if left.resultdef.typ=stringdef then
|
|
begin
|
|
result:=nil;
|
|
end
|
|
{$endif}
|
|
else
|
|
result:=inherited first_length;
|
|
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
|
|
{$ifndef nounsupported}
|
|
else if left.resultdef.typ=stringdef then
|
|
begin
|
|
location_reset(location,LOC_REGISTER,OS_S32);
|
|
location.register:=hlcg.getintregister(current_asmdata.CurrAsmList,s32inttype);
|
|
thlcgjvm(hlcg).a_load_const_reg(current_asmdata.CurrAsmList,s32inttype,0,location.register);
|
|
end
|
|
{$endif}
|
|
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;
|
|
emptystr: ansichar;
|
|
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_wide_or_unicode_string(target.resultdef) then
|
|
begin
|
|
emptystr:=#0;
|
|
current_asmdata.CurrAsmList.concat(taicpu.op_string(a_ldc,0,@emptystr));
|
|
thlcgjvm(hlcg).incstack(current_asmdata.CurrAsmList,1);
|
|
end
|
|
else 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
|
|
{$ifndef nounsupported}
|
|
else if left.resultdef.typ=stringdef then
|
|
begin
|
|
thlcgjvm(hlcg).a_load_const_stack(current_asmdata.CurrAsmList,java_jlobject,0,R_ADDRESSREGISTER);
|
|
end
|
|
{$endif}
|
|
else
|
|
internalerror(2011031401);
|
|
thlcgjvm(hlcg).a_load_stack_loc(current_asmdata.CurrAsmList,target.resultdef,target.location);
|
|
end;
|
|
|
|
procedure tjvminlinenode.second_box;
|
|
begin
|
|
{$ifndef nounsupported}
|
|
secondpass(tcallparanode(left).left);
|
|
location_reset(location,LOC_REGISTER,OS_ADDR);
|
|
location.register:=hlcg.getaddressregister(current_asmdata.CurrAsmList,java_jlobject);
|
|
hlcg.a_load_const_reg(current_asmdata.CurrAsmList,java_jlobject,0,location.register);
|
|
{$else}
|
|
internalerror(2011042606);
|
|
{$endif}
|
|
end;
|
|
|
|
|
|
begin
|
|
cinlinenode:=tjvminlinenode;
|
|
end.
|