| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493 | {    Copyright (c) 1998-2011 by Florian Klaempfl and Jonas Maebe    Generate assembler for constant nodes for the JVM    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 njvmcon;{$i fpcdefs.inc}interface    uses       globtype,aasmbase,       symtype,       node,ncal,ncon,ncgcon;    type       tjvmordconstnode = class(tcgordconstnode)          { normally, we convert the enum constant into a load of the            appropriate enum class field in pass_1. In some cases (array index),            we want to keep it as an enum constant however }          enumconstok: boolean;          function pass_1: tnode; override;          function docompare(p: tnode): boolean; override;          function dogetcopy: tnode; override;       end;       tjvmrealconstnode = class(tcgrealconstnode)          procedure pass_generate_code;override;       end;       tjvmstringconstnode = class(tstringconstnode)          function pass_1: tnode; override;          procedure pass_generate_code;override;          class function emptydynstrnil: boolean; override;       end;       tjvmsetconsttype = (         { create symbol for the set constant; the symbol will be initialized           in the class constructor/unit init code (default) }         sct_constsymbol,         { normally, we convert the set constant into a constructor/factory           method to create a set instance. In some cases (simple "in"           expressions, adding an element to an empty set, ...) we want to           keep the set constant instead }         sct_notransform,         { actually construct a JUBitSet/JUEnumSet that contains the set value           (for initializing the sets contstants) }         sct_construct         );       tjvmsetconstnode = class(tcgsetconstnode)          setconsttype: tjvmsetconsttype;          function pass_1: tnode; override;          procedure pass_generate_code; override;          constructor create(s : pconstset;def:tdef);override;          function docompare(p: tnode): boolean; override;          function dogetcopy: tnode; override;         protected          function emitvarsetconst: tasmsymbol; override;          { in case the set has only a single run of consecutive elements,            this function will return its starting index and length }          function find_single_elements_run(from: longint; out start, len: longint): boolean;          function buildbitset: tnode;          function buildenumset(const eledef: tdef): tnode;          function buildsetfromstring(const helpername: string; otherparas: tcallparanode): tnode;       end;implementation    uses      globals,cutils,widestr,verbose,constexp,fmodule,      symdef,symsym,symcpu,symtable,symconst,      aasmdata,aasmcpu,defutil,      nutils,ncnv,nld,nmem,pjvm,pass_1,      cgbase,hlcgobj,hlcgcpu,cgutils,cpubase      ;{*****************************************************************************                           TJVMORDCONSTNODE*****************************************************************************}    function tjvmordconstnode.pass_1: tnode;      var        basedef: tcpuenumdef;        sym: tenumsym;        classfield: tsym;      begin        if (resultdef.typ<>enumdef) or           enumconstok then          begin            result:=inherited pass_1;            exit;          end;        { convert into JVM class instance }        { a) find the enumsym corresponding to the value (may not exist in case             of an explicit typecast of an integer -> error) }        sym:=nil;        sym:=tenumsym(tenumdef(resultdef).int2enumsym(int64(value)));        if not assigned(sym) then          begin            Message(parser_e_range_check_error);            result:=nil;            exit;          end;        { b) find the corresponding class field }        basedef:=tcpuenumdef(tenumdef(resultdef).getbasedef);        classfield:=search_struct_member(basedef.classdef,sym.name);        { c) create loadnode of the field }        result:=nil;        if not handle_staticfield_access(classfield,result) then          internalerror(2011062606);      end;    function tjvmordconstnode.docompare(p: tnode): boolean;      begin        result:=inherited docompare(p);        if result then          result:=(enumconstok=tjvmordconstnode(p).enumconstok);      end;    function tjvmordconstnode.dogetcopy: tnode;      begin        result:=inherited dogetcopy;        tjvmordconstnode(result).enumconstok:=enumconstok;      end;{*****************************************************************************                           TJVMREALCONSTNODE*****************************************************************************}    procedure tjvmrealconstnode.pass_generate_code;      begin        location_reset(location,LOC_FPUREGISTER,def_cgsize(resultdef));        location.register:=hlcg.getfpuregister(current_asmdata.CurrAsmList,resultdef);        thlcgjvm(hlcg).a_loadfpu_const_stack(current_asmdata.CurrAsmList,resultdef,value_real);        thlcgjvm(hlcg).a_load_stack_reg(current_asmdata.CurrAsmList,resultdef,location.register);      end;    { tcgstringconstnode }    function tjvmstringconstnode.pass_1: tnode;      var        strclass: tobjectdef;        pw: pcompilerwidestring;        paras: tcallparanode;        wasansi: boolean;      begin        { all Java strings are utf-16. However, there is no way to          declare a constant array of bytes (or any other type), those          have to be constructed by declaring a final field and then          initialising them in the class constructor element per          element. We therefore put the straight ASCII values into          the UTF-16 string, and then at run time extract those and          store them in an Ansistring/AnsiChar array }        result:=inherited pass_1;        if assigned(result) or           (cst_type in [cst_unicodestring,cst_widestring]) then          exit;        { convert the constant into a widestring representation without any          code page conversion }        initwidestring(pw);        ascii2unicode(value_str,len,current_settings.sourcecodepage,pw,false);        ansistringdispose(value_str,len);        pcompilerwidestring(value_str):=pw;        { and now add a node to convert the data into ansistring format at          run time }        wasansi:=false;        case cst_type of          cst_ansistring:            begin              if len=0 then                begin                  { we have to use nil rather than an empty string, because an                    empty string has a code page and this messes up the code                    page selection logic in the RTL }                  exit;                end;              strclass:=tobjectdef(search_system_type('ANSISTRINGCLASS').typedef);              wasansi:=true;            end;          cst_shortstring:            strclass:=tobjectdef(search_system_type('SHORTSTRINGCLASS').typedef);          cst_conststring:            { used for array of char }            strclass:=tobjectdef(search_system_type('ANSICHARARRAYCLASS').typedef);          else           internalerror(2011052401);        end;        cst_type:=cst_unicodestring;        paras:=ccallparanode.create(self.getcopy,nil);        if wasansi then          paras:=ccallparanode.create(            genintconstnode(tstringdef(resultdef).encoding),paras);        { since self will be freed, have to make a copy }        result:=ccallnode.createinternmethodres(          cloadvmtaddrnode.create(ctypenode.create(strclass)),          'CREATEFROMLITERALSTRINGBYTES',paras,resultdef);      end;    procedure tjvmstringconstnode.pass_generate_code;      begin        location_reset(location,LOC_REGISTER,def_cgsize(resultdef));        location.register:=hlcg.getaddressregister(current_asmdata.CurrAsmList,resultdef);        case cst_type of          cst_ansistring:            begin              if len<>0 then                internalerror(2012052604);              hlcg.a_load_const_reg(current_asmdata.CurrAsmList,resultdef,0,location.register);              { done }              exit;            end;          cst_shortstring,          cst_conststring:            internalerror(2012052601);          cst_unicodestring,          cst_widestring:            current_asmdata.CurrAsmList.concat(taicpu.op_wstring(a_ldc,pcompilerwidestring(value_str)));          else            internalerror(2012052602);        end;        thlcgjvm(hlcg).incstack(current_asmdata.CurrAsmList,1);        thlcgjvm(hlcg).a_load_stack_reg(current_asmdata.CurrAsmList,resultdef,location.register);      end;    class function tjvmstringconstnode.emptydynstrnil: boolean;      begin        result:=false;      end;    {*****************************************************************************                               TJVMSETCONSTNODE    *****************************************************************************}    function tjvmsetconstnode.buildsetfromstring(const helpername: string; otherparas: tcallparanode): tnode;      var        pw: pcompilerwidestring;        wc: tcompilerwidechar;        i, j, bit, nulls: longint;      begin        initwidestring(pw);        nulls:=0;        for i:=0 to 15 do          begin            wc:=0;            for bit:=0 to 15 do              if (i*16+bit) in value_set^ then                wc:=wc or (1 shl (15-bit));            { don't add trailing zeroes }            if wc=0 then              inc(nulls)            else              begin                for j:=1 to nulls do                  concatwidestringchar(pw,0);                nulls:=0;                concatwidestringchar(pw,wc);              end;          end;        result:=ccallnode.createintern(helpername,          ccallparanode.create(cstringconstnode.createunistr(pw),otherparas));        donewidestring(pw);      end;    function tjvmsetconstnode.buildbitset: tnode;      var        mp: tnode;      begin        if value_set^=[] then          begin            mp:=cloadvmtaddrnode.create(ctypenode.create(java_jubitset));            result:=ccallnode.createinternmethod(mp,'CREATE',nil);            exit;          end;        result:=buildsetfromstring('fpc_bitset_from_string',nil);      end;    function tjvmsetconstnode.buildenumset(const eledef: tdef): tnode;      var        stopnode: tnode;        startnode: tnode;        mp: tnode;        len: longint;        start: longint;        enumele: tnode;        paras: tcallparanode;        hassinglerun: boolean;      begin        hassinglerun:=find_single_elements_run(0, start, len);        if hassinglerun then          begin            mp:=cloadvmtaddrnode.create(ctypenode.create(java_juenumset));            if len=0 then              begin                enumele:=cloadvmtaddrnode.create(ctypenode.create(tcpuenumdef(tenumdef(eledef).getbasedef).classdef));                inserttypeconv_explicit(enumele,search_system_type('JLCLASS').typedef);                paras:=ccallparanode.create(enumele,nil);                result:=ccallnode.createinternmethod(mp,'NONEOF',paras)              end            else              begin                startnode:=cordconstnode.create(start,eledef,false);                { immediately firstpass so the enum gets translated into a JLEnum                  instance }                firstpass(startnode);                if len=1 then                  result:=ccallnode.createinternmethod(mp,'OF',ccallparanode.create(startnode,nil))                else                  begin                    stopnode:=cordconstnode.create(start+len-1,eledef,false);                    firstpass(stopnode);                    result:=ccallnode.createinternmethod(mp,'RANGE',ccallparanode.create(stopnode,ccallparanode.create(startnode,nil)));                  end              end          end        else          begin            enumele:=cordconstnode.create(tenumsym(tenumdef(eledef).symtable.symlist[0]).value,eledef,false);            firstpass(enumele);            paras:=ccallparanode.create(enumele,nil);            result:=buildsetfromstring('fpc_enumset_from_string',paras);          end;      end;    function tjvmsetconstnode.pass_1: tnode;      var        eledef: tdef;      begin        { we want set constants to be global, so we can reuse them. However,          if the set's elementdef is local, we can't do that since a global          symbol cannot have a local definition (the compiler will crash when          loading the ppu file afterwards) }        if tsetdef(resultdef).elementdef.owner.symtabletype=localsymtable then          setconsttype:=sct_construct;        result:=nil;        case setconsttype of(*          sct_constsymbol:            begin              { normally a codegen pass routine, but we have to insert a typed                const in case the set constant does not exist yet, and that                should happen in pass_1 (especially since it involves creating                new nodes, which may even have to be tacked on to this code in                case it's the unit initialization code) }              handlevarsetconst;              { no smallsets }              expectloc:=LOC_CREFERENCE;            end;*)          sct_notransform:            begin              result:=inherited pass_1;              { no smallsets }              expectloc:=LOC_CREFERENCE;            end;          sct_constsymbol,          sct_construct:            begin              eledef:=tsetdef(resultdef).elementdef;              { empty sets don't have an element type, so we don't know whether we                have to constructor a bitset or enumset (and of which type) }              if not assigned(eledef) then                internalerror(2011070202);              if eledef.typ=enumdef then                begin                  result:=buildenumset(eledef);                end              else                begin                  result:=buildbitset;                end;              inserttypeconv_explicit(result,cpointerdef.getreusable(resultdef));              result:=cderefnode.create(result);            end;        end;      end;    procedure tjvmsetconstnode.pass_generate_code;      begin        case setconsttype of          sct_constsymbol:            begin              { all sets are varsets for the JVM target, no setbase differences }              handlevarsetconst;            end;          else            { must be handled in pass_1 or otherwise transformed }            internalerror(2011070201)        end;      end;    constructor tjvmsetconstnode.create(s: pconstset; def: tdef);      begin        inherited create(s, def);        setconsttype:=sct_constsymbol;      end;    function tjvmsetconstnode.docompare(p: tnode): boolean;      begin        result:=          inherited docompare(p) and          (setconsttype=tjvmsetconstnode(p).setconsttype);      end;    function tjvmsetconstnode.dogetcopy: tnode;      begin        result:=inherited dogetcopy;        tjvmsetconstnode(result).setconsttype:=setconsttype;      end;    function tjvmsetconstnode.emitvarsetconst: tasmsymbol;      var        csym: tconstsym;        ssym: tstaticvarsym;        ps: pnormalset;      begin        { add a read-only typed constant }        new(ps);        ps^:=value_set^;        csym:=cconstsym.create_ptr('_$setconst'+tostr(current_module.symlist.count),constset,ps,resultdef);        csym.visibility:=vis_private;        include(csym.symoptions,sp_internal);        current_module.localsymtable.insertsym(csym);        { generate assignment of the constant to the typed constant symbol }        ssym:=jvm_add_typed_const_initializer(csym);        result:=current_asmdata.RefAsmSymbol(ssym.mangledname,AT_DATA);      end;    function tjvmsetconstnode.find_single_elements_run(from: longint; out start, len: longint): boolean;      var        i: longint;      begin        i:=from;        result:=true;        { find first element in set }        while (i<=255) and              not(i in value_set^) do          inc(i);        start:=i;        { go to end of the run }        while (i<=255) and              (i in value_set^) do          inc(i);        len:=i-start;        { rest must be unset }        while (i<=255) and              not(i in value_set^) do          inc(i);        if i<>256 then          result:=false;      end;begin   cordconstnode:=tjvmordconstnode;   crealconstnode:=tjvmrealconstnode;   cstringconstnode:=tjvmstringconstnode;   csetconstnode:=tjvmsetconstnode;end.
 |