mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-12 14:49:33 +02:00

o support for copying value parameters at the callee side if they were passed by reference in hlcg o JVM g_concatcopy() implementation for arrays o moved code to get length of an array from njvminl to hlcgcpu so it can be reused elsewhere as well o export array copy helpers from system unit for use when assigning one array to another o some generic support for types that are normally not implicit pointers, but which are for the JVM target (such as normal arrays) * handle assigning nil to a dynamic array by generating a setlength(x,0) node instead of by hardcoding a call to fpc_dynarray_clear, so target-specific code can handle it if required * hook up gethltemp() for JVM ttgjvm so array temps are properly allocated git-svn-id: branches/jvmbackend@18388 -
122 lines
4.8 KiB
ObjectPascal
122 lines
4.8 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;
|
|
|
|
type
|
|
tjvmvecnode = class(tcgvecnode)
|
|
procedure pass_generate_code;override;
|
|
end;
|
|
|
|
implementation
|
|
|
|
uses
|
|
systems,
|
|
cutils,verbose,
|
|
symdef,defutil,
|
|
aasmdata,pass_2,
|
|
cgutils,hlcgobj,hlcgcpu;
|
|
|
|
{*****************************************************************************
|
|
TJVMVECNODE
|
|
*****************************************************************************}
|
|
|
|
procedure tjvmvecnode.pass_generate_code;
|
|
var
|
|
newsize: tcgsize;
|
|
begin
|
|
{ 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);
|
|
|
|
{ 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
|
|
cvecnode:=tjvmvecnode;
|
|
end.
|