mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-28 19:33:50 +02:00

them to shortstringclass, make sure the address operator is "typed" so that we don't insert useless checkcast instructions git-svn-id: branches/jvmbackend@18621 -
353 lines
14 KiB
ObjectPascal
353 lines
14 KiB
ObjectPascal
{
|
|
Copyright (c) 2011 by Jonas Maebe
|
|
|
|
Generate JVM byetcode for in memory related 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 njvmmem;
|
|
|
|
{$i fpcdefs.inc}
|
|
|
|
interface
|
|
|
|
uses
|
|
globtype,
|
|
cgbase,cpubase,
|
|
node,nmem,ncgmem,ncgnstmm;
|
|
|
|
type
|
|
tjvmaddrnode = class(tcgaddrnode)
|
|
function pass_typecheck: tnode; override;
|
|
procedure pass_generate_code; override;
|
|
end;
|
|
|
|
tjvmderefnode = class(tcgderefnode)
|
|
function pass_typecheck:tnode;override;
|
|
procedure pass_generate_code; override;
|
|
end;
|
|
|
|
tjvmloadvmtaddrnode = class(tcgloadvmtaddrnode)
|
|
procedure pass_generate_code; override;
|
|
end;
|
|
|
|
tjvmvecnode = class(tcgvecnode)
|
|
function pass_1: tnode; override;
|
|
procedure pass_generate_code;override;
|
|
end;
|
|
|
|
implementation
|
|
|
|
uses
|
|
systems,globals,
|
|
cutils,verbose,constexp,
|
|
symconst,symtype,symtable,symsym,symdef,defutil,jvmdef,
|
|
htypechk,
|
|
nadd,ncal,ncnv,ncon,pass_1,njvmcon,
|
|
aasmdata,aasmcpu,pass_2,
|
|
cgutils,hlcgobj,hlcgcpu;
|
|
|
|
{*****************************************************************************
|
|
TJVMDEREFNODE
|
|
*****************************************************************************}
|
|
|
|
function tjvmderefnode.pass_typecheck: tnode;
|
|
begin
|
|
result:=inherited;
|
|
if not(left.resultdef.typ=pointerdef) or
|
|
((left.resultdef<>voidpointertype) and
|
|
not jvmimplicitpointertype(tpointerdef(left.resultdef).pointeddef)) then
|
|
begin
|
|
CGMessage(parser_e_illegal_expression);
|
|
exit
|
|
end;
|
|
end;
|
|
|
|
procedure tjvmderefnode.pass_generate_code;
|
|
var
|
|
implicitptr: boolean;
|
|
begin
|
|
secondpass(left);
|
|
implicitptr:=jvmimplicitpointertype(tpointerdef(left.resultdef).pointeddef);
|
|
if (left.resultdef.typ=pointerdef) and
|
|
((left.resultdef=voidpointertype) or
|
|
implicitptr) then
|
|
begin
|
|
if implicitptr then
|
|
{ this is basically a typecast: the left node is a regular
|
|
'pointer', and we typecast it to an implicit pointer }
|
|
location_copy(location,left.location)
|
|
else
|
|
begin
|
|
{ these are always arrays (used internally for pointers to var
|
|
parameters stored in nestedfpstructs) }
|
|
hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,left.resultdef,left.resultdef,true);
|
|
location_reset_ref(location,LOC_REFERENCE,OS_ADDR,4);
|
|
reference_reset_base(location.reference,left.location.register,0,4);
|
|
location.reference.arrayreftype:=art_indexconst;
|
|
end;
|
|
end
|
|
else
|
|
internalerror(2011052901);
|
|
end;
|
|
|
|
{*****************************************************************************
|
|
TJVMADDRNODE
|
|
*****************************************************************************}
|
|
|
|
function tjvmaddrnode.pass_typecheck: tnode;
|
|
begin
|
|
result:=nil;
|
|
typecheckpass(left);
|
|
if codegenerror then
|
|
exit;
|
|
|
|
make_not_regable(left,[ra_addr_regable,ra_addr_taken]);
|
|
|
|
if (left.resultdef.typ=procdef) or
|
|
(
|
|
(left.resultdef.typ=procvardef) and
|
|
((m_tp_procvar in current_settings.modeswitches) or
|
|
(m_mac_procvar in current_settings.modeswitches))
|
|
) then
|
|
begin
|
|
result:=inherited;
|
|
exit;
|
|
end
|
|
else
|
|
begin
|
|
if not(nf_internal in flags) and
|
|
not jvmimplicitpointertype(left.resultdef) then
|
|
begin
|
|
CGMessage(parser_e_illegal_expression);
|
|
exit
|
|
end;
|
|
result:=inherited;
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure tjvmaddrnode.pass_generate_code;
|
|
var
|
|
implicitptr: boolean;
|
|
begin
|
|
secondpass(left);
|
|
implicitptr:=jvmimplicitpointertype(left.resultdef);
|
|
if implicitptr or
|
|
(nf_internal in flags) then
|
|
begin
|
|
if implicitptr then
|
|
{ this is basically a typecast: the left node is an implicit
|
|
pointer, and we typecast it to a regular 'pointer'
|
|
(java.lang.Object) }
|
|
location_copy(location,left.location)
|
|
else
|
|
begin
|
|
{ these are always arrays (used internally for pointers to var
|
|
parameters stored in nestedfpstructs) -> get base pointer to
|
|
array }
|
|
if (left.location.loc<>LOC_REFERENCE) or
|
|
(left.location.reference.arrayreftype<>art_indexconst) or
|
|
(left.location.reference.base=NR_NO) or
|
|
assigned(left.location.reference.symbol) then
|
|
internalerror(2011060701);
|
|
location_reset(location,LOC_REGISTER,OS_ADDR);
|
|
location.register:=left.location.reference.base;
|
|
end;
|
|
end
|
|
else
|
|
begin
|
|
{ procvar }
|
|
{$ifndef nounsupported}
|
|
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(2011051601);
|
|
{$endif}
|
|
end;
|
|
end;
|
|
|
|
{*****************************************************************************
|
|
TJVMLOADVMTADDRNODE
|
|
*****************************************************************************}
|
|
|
|
procedure tjvmloadvmtaddrnode.pass_generate_code;
|
|
begin
|
|
current_asmdata.CurrAsmList.concat(taicpu.op_sym(a_ldc,current_asmdata.RefAsmSymbol(
|
|
tobjectdef(tclassrefdef(resultdef).pointeddef).jvm_full_typename(true))));
|
|
thlcgjvm(hlcg).incstack(current_asmdata.CurrAsmList,1);
|
|
location_reset(location,LOC_REGISTER,OS_ADDR);
|
|
location.register:=hlcg.getaddressregister(current_asmdata.CurrAsmList,resultdef);
|
|
thlcgjvm(hlcg).a_load_stack_reg(current_asmdata.CurrAsmList,resultdef,location.register);
|
|
end;
|
|
|
|
|
|
{*****************************************************************************
|
|
TJVMVECNODE
|
|
*****************************************************************************}
|
|
|
|
function tjvmvecnode.pass_1: tnode;
|
|
var
|
|
psym: tsym;
|
|
stringclass: tdef;
|
|
begin
|
|
if (left.resultdef.typ=stringdef) then
|
|
begin
|
|
case tstringdef(left.resultdef).stringtype of
|
|
st_ansistring:
|
|
stringclass:=java_ansistring;
|
|
st_unicodestring,
|
|
st_widestring:
|
|
stringclass:=java_jlstring;
|
|
st_shortstring:
|
|
begin
|
|
stringclass:=java_shortstring;
|
|
left:=caddrnode.create_internal(left);
|
|
{ avoid useless typecheck when casting to shortstringclass }
|
|
include(left.flags,nf_typedaddr);
|
|
end
|
|
else
|
|
internalerror(2011052407);
|
|
end;
|
|
psym:=search_struct_member(tabstractrecorddef(stringclass),'CHARAT');
|
|
if not assigned(psym) or
|
|
(psym.typ<>procsym) then
|
|
internalerror(2011031501);
|
|
{ Pascal strings are 1-based, Java strings 0-based }
|
|
result:=ccallnode.create(ccallparanode.create(
|
|
caddnode.create(subn,right,genintconstnode(1)),nil),tprocsym(psym),
|
|
psym.owner,ctypeconvnode.create_explicit(left,stringclass),[]);
|
|
left:=nil;
|
|
right:=nil;
|
|
exit;
|
|
end
|
|
else
|
|
begin
|
|
{ keep indices that are enum constants that way, rather than
|
|
transforming them into a load of the class instance that
|
|
represents this constant (since we then would have to extract
|
|
the int constant value again at run time anyway) }
|
|
if right.nodetype=ordconstn then
|
|
tjvmordconstnode(right).enumconstok:=true;
|
|
result:=inherited;
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure tjvmvecnode.pass_generate_code;
|
|
var
|
|
psym: tsym;
|
|
newsize: tcgsize;
|
|
begin
|
|
if left.resultdef.typ=stringdef then
|
|
internalerror(2011052702);
|
|
|
|
{ This routine is not used for Strings, as they are a class type and
|
|
you have to use charAt() there to load a character (and you cannot
|
|
change characters; you have to create a new string in that case)
|
|
|
|
As far as arrays are concerned: we have to create a trefererence
|
|
with arrayreftype in [art_indexreg,art_indexref], and ref.base =
|
|
pointer to the array (i.e., left.location.register) }
|
|
secondpass(left);
|
|
newsize:=def_cgsize(resultdef);
|
|
if left.location.loc=LOC_CREFERENCE then
|
|
location_reset_ref(location,LOC_CREFERENCE,newsize,left.location.reference.alignment)
|
|
else
|
|
location_reset_ref(location,LOC_REFERENCE,newsize,left.location.reference.alignment);
|
|
{ don't use left.resultdef, because it may be an open or regular array,
|
|
and then asking for the size doesn't make any sense }
|
|
hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,java_jlobject,java_jlobject,true);
|
|
location.reference.base:=left.location.register;
|
|
secondpass(right);
|
|
{ simplify index location if necessary, since array references support
|
|
an index in memory, but not an another array index }
|
|
if (right.location.loc in [LOC_REFERENCE,LOC_CREFERENCE]) and
|
|
(right.location.reference.arrayreftype<>art_none) then
|
|
hlcg.location_force_reg(current_asmdata.CurrAsmList,right.location,right.resultdef,right.resultdef,true);
|
|
{ replace enum class instance with the corresponding integer value }
|
|
if (right.resultdef.typ=enumdef) then
|
|
begin
|
|
if (right.location.loc<>LOC_CONSTANT) then
|
|
begin
|
|
psym:=search_struct_member(tenumdef(right.resultdef).classdef,'FPCORDINAL');
|
|
if not assigned(psym) or
|
|
(psym.typ<>procsym) or
|
|
(tprocsym(psym).ProcdefList.count<>1) then
|
|
internalerror(2011062607);
|
|
thlcgjvm(hlcg).a_load_loc_stack(current_asmdata.CurrAsmList,right.resultdef,right.location);
|
|
hlcg.a_call_name(current_asmdata.CurrAsmList,tprocdef(tprocsym(psym).procdeflist[0]),tprocdef(tprocsym(psym).procdeflist[0]).mangledname,false);
|
|
{ call replaces self parameter with longint result -> no stack
|
|
height change }
|
|
location_reset(right.location,LOC_REGISTER,OS_S32);
|
|
right.location.register:=hlcg.getintregister(current_asmdata.CurrAsmList,s32inttype);
|
|
thlcgjvm(hlcg).a_load_stack_reg(current_asmdata.CurrAsmList,s32inttype,right.location.register);
|
|
end;
|
|
{ always force to integer location, because enums are handled as
|
|
object instances (since that's what they are in Java) }
|
|
right.resultdef:=s32inttype;
|
|
right.location.size:=OS_S32;
|
|
end;
|
|
|
|
{ adjust index if necessary }
|
|
if not is_special_array(left.resultdef) and
|
|
(tarraydef(left.resultdef).lowrange<>0) and
|
|
(right.location.loc<>LOC_CONSTANT) then
|
|
begin
|
|
thlcgjvm(hlcg).a_load_loc_stack(current_asmdata.CurrAsmList,right.resultdef,right.location);
|
|
thlcgjvm(hlcg).a_op_const_stack(current_asmdata.CurrAsmList,OP_SUB,right.resultdef,tarraydef(left.resultdef).lowrange);
|
|
if right.location.loc<>LOC_REGISTER then
|
|
begin
|
|
location_reset(right.location,LOC_REGISTER,def_cgsize(right.resultdef));
|
|
right.location.register:=hlcg.getintregister(current_asmdata.CurrAsmList,right.resultdef);
|
|
end;
|
|
thlcgjvm(hlcg).a_load_stack_reg(current_asmdata.CurrAsmList,right.resultdef,right.location.register);
|
|
end;
|
|
|
|
{ create array reference }
|
|
case right.location.loc of
|
|
LOC_REGISTER,LOC_CREGISTER:
|
|
begin
|
|
location.reference.arrayreftype:=art_indexreg;
|
|
location.reference.index:=right.location.register;
|
|
end;
|
|
LOC_REFERENCE,LOC_CREFERENCE:
|
|
begin
|
|
location.reference.arrayreftype:=art_indexref;
|
|
location.reference.indexbase:=right.location.reference.base;
|
|
location.reference.indexsymbol:=right.location.reference.symbol;
|
|
location.reference.indexoffset:=right.location.reference.offset;
|
|
end;
|
|
LOC_CONSTANT:
|
|
begin
|
|
location.reference.arrayreftype:=art_indexconst;
|
|
location.reference.indexoffset:=right.location.value-tarraydef(left.resultdef).lowrange;
|
|
end
|
|
else
|
|
internalerror(2011012002);
|
|
end;
|
|
end;
|
|
|
|
|
|
begin
|
|
cderefnode:=tjvmderefnode;
|
|
caddrnode:=tjvmaddrnode;
|
|
cvecnode:=tjvmvecnode;
|
|
cloadvmtaddrnode:=tjvmloadvmtaddrnode;
|
|
end.
|