fpc/compiler/jvm/njvmmem.pas
Jonas Maebe bf2854dae5 * when internally taking the address of shortstrings to typecast
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 -
2011-08-20 08:15:59 +00:00

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.