| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096 | {    Copyright (c) 1998-2002 by Florian Klaempfl    Generate assembler for memory related nodes which are    the same for all (most?) processors    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 ncgmem;{$i fpcdefs.inc}interface    uses      globtype,cgbase,cgutils,cpubase,      symtype,      node,nmem;    type       tcgloadvmtaddrnode = class(tloadvmtaddrnode)          procedure pass_generate_code;override;       end;       tcgloadparentfpnode = class(tloadparentfpnode)          procedure pass_generate_code;override;       end;       tcgaddrnode = class(taddrnode)          procedure pass_generate_code;override;       end;       tcgderefnode = class(tderefnode)          procedure pass_generate_code;override;       end;       tcgsubscriptnode = class(tsubscriptnode)         protected          function handle_platform_subscript: boolean; virtual;         public          procedure pass_generate_code;override;       end;       tcgvecnode = class(tvecnode)         function get_mul_size : asizeint;       private         procedure rangecheck_array;         procedure rangecheck_string;       protected         {# This routine is used to calculate the address of the reference.            On entry reg contains the index in the array,           and l contains the size of each element in the array.           This routine should update location.reference correctly,           so it points to the correct address.         }         procedure update_reference_reg_mul(maybe_const_reg: tregister;regsize: tdef; l: aint);virtual;         procedure update_reference_reg_packed(maybe_const_reg: tregister; regsize: tdef; l: aint);virtual;         procedure update_reference_offset(var ref: treference; index, mulsize: ASizeInt); virtual;         procedure second_wideansistring;virtual;         procedure second_dynamicarray;virtual;         function valid_index_size(size: tcgsize): boolean;virtual;       public         procedure pass_generate_code;override;       end;implementation    uses      systems,      cutils,cclasses,verbose,globals,constexp,fmodule,      symconst,symbase,symdef,symsym,symtable,defutil,paramgr,      aasmbase,aasmdata,      procinfo,pass_2,parabase,      ncon,nadd,nutils,      cgobj,hlcgobj,      objcgutl;{*****************************************************************************                              TCGLOADVMTADDRNODE*****************************************************************************}    procedure tcgloadvmtaddrnode.pass_generate_code;      var        href    : treference;        pool    : THashSet;        entry   : PHashSetItem;        vmtname : tsymstr;        otherunit,        indirect : boolean;      begin         location_reset(location,LOC_REGISTER,def_cgsize(voidpointertype));         if (left.nodetype=typen) then           begin             location.register:=hlcg.getaddressregister(current_asmdata.CurrAsmList,voidpointertype);             if not is_objcclass(left.resultdef) then               begin                 { we are using a direct reference if any of the following is true:                   - the target does not support packages                   - the target does not use indirect references                   - the class is located inside the same unit }                 otherunit:=findunitsymtable(left.resultdef.owner).moduleid<>current_module.moduleid;                 indirect:=(tf_supports_packages in target_info.flags) and                           (target_info.system in systems_indirect_var_imports) and                           otherunit;                 vmtname:=tobjectdef(tclassrefdef(resultdef).pointeddef).vmt_mangledname;                 reference_reset_symbol(href,                   current_asmdata.RefAsmSymbol(vmtname,AT_DATA,indirect),0,                   resultdef.alignment,[]);                 hlcg.a_loadaddr_ref_reg(current_asmdata.CurrAsmList,resultdef,resultdef,href,location.register);                 if otherunit then                   current_module.add_extern_asmsym(vmtname,AB_EXTERNAL,AT_DATA);               end             else               begin                 pool:=current_asmdata.ConstPools[sp_objcclassnamerefs];                 entry:=pool.FindOrAdd(@tobjectdef(left.resultdef).objextname^[1],length(tobjectdef(left.resultdef).objextname^));                 if (target_info.system in systems_objc_nfabi) then                   begin                     { find/add necessary classref/classname pool entries }                     objcfinishclassrefnfpoolentry(entry,tobjectdef(left.resultdef));                   end                 else                   begin                     { find/add necessary classref/classname pool entries }                     objcfinishstringrefpoolentry(entry,sp_objcclassnames,sec_objc_cls_refs,sec_objc_class_names);                   end;                 reference_reset_symbol(href,tasmlabel(entry^.Data),0,objc_idtype.alignment,[]);                 hlcg.a_load_ref_reg(current_asmdata.CurrAsmList,objc_idtype,objc_idtype,href,location.register);               end;           end         else           { should be handled in pass 1 }           internalerror(2015052801);      end;{*****************************************************************************                        TCGLOADPARENTFPNODE*****************************************************************************}    procedure tcgloadparentfpnode.pass_generate_code;      var        currpi : tprocinfo;        hsym   : tparavarsym;        href   : treference;      begin        if (current_procinfo.procdef.parast.symtablelevel=parentpd.parast.symtablelevel) then          begin            location_reset(location,LOC_REGISTER,def_cgsize(parentfpvoidpointertype));            location.register:=current_procinfo.framepointer;          end        else          begin            currpi:=current_procinfo;            location_reset(location,LOC_REGISTER,def_cgsize(parentfpvoidpointertype));            location.register:=hlcg.getaddressregister(current_asmdata.CurrAsmList,parentfpvoidpointertype);            { load framepointer of current proc }            hsym:=tparavarsym(currpi.procdef.parast.Find('parentfp'));            if not assigned(hsym) then              internalerror(200309281);            hlcg.a_load_loc_reg(current_asmdata.CurrAsmList,parentfpvoidpointertype,parentfpvoidpointertype,hsym.localloc,location.register);            { walk parents }            while (currpi.procdef.owner.symtablelevel>parentpd.parast.symtablelevel) do              begin                currpi:=currpi.parent;                if not assigned(currpi) then                  internalerror(200311201);                hsym:=tparavarsym(currpi.procdef.parast.Find('parentfp'));                if not assigned(hsym) then                  internalerror(200309282);                if hsym.localloc.loc<>LOC_REFERENCE then                  internalerror(200309283);                hlcg.reference_reset_base(href,parentfpvoidpointertype,location.register,hsym.localloc.reference.offset,ctempposinvalid,parentfpvoidpointertype.alignment,[]);                hlcg.a_load_ref_reg(current_asmdata.CurrAsmList,parentfpvoidpointertype,parentfpvoidpointertype,href,location.register);              end;          end;      end;{*****************************************************************************                             TCGADDRNODE*****************************************************************************}    procedure tcgaddrnode.pass_generate_code;      begin         secondpass(left);         location_reset(location,LOC_REGISTER,int_cgsize(resultdef.size));         location.register:=hlcg.getaddressregister(current_asmdata.CurrAsmList,resultdef);         if not(left.location.loc in [LOC_REFERENCE,LOC_CREFERENCE]) then           { on x86_64-win64, array of chars can be returned in registers, however,             when passing these arrays to other functions, the compiler wants to take             the address of the array so when the addrnode has been created internally,             we have to force the data into memory, see also tw14388.pp           }           if nf_internal in flags then             hlcg.location_force_mem(current_asmdata.CurrAsmList,left.location,left.resultdef)           else             internalerror(2006111510);         hlcg.a_loadaddr_ref_reg(current_asmdata.CurrAsmList,left.resultdef,resultdef,left.location.reference,location.register);      end;{*****************************************************************************                           TCGDEREFNODE*****************************************************************************}    procedure tcgderefnode.pass_generate_code;      var        paraloc1 : tcgpara;        pd : tprocdef;        sym : tsym;        st : tsymtable;        hp : pnode;        extraoffset : tcgint;      begin         sym:=nil;         { assume natural alignment, except for packed records }         if not(resultdef.typ in [recorddef,objectdef]) or            (tabstractrecordsymtable(tabstractrecorddef(resultdef).symtable).usefieldalignment<>1) then           location_reset_ref(location,LOC_REFERENCE,def_cgsize(resultdef),resultdef.alignment,[])         else           location_reset_ref(location,LOC_REFERENCE,def_cgsize(resultdef),1,[]);         { can we fold an add/sub node into the offset of the deref node? }         extraoffset:=0;         hp:=actualtargetnode(@left);         if (hp^.nodetype=subn) and is_constintnode(taddnode(hp^).right) then           begin             extraoffset:=-tcgint(tordconstnode(taddnode(hp^).right).value);             replacenode(hp^,taddnode(hp^).left);           end         else if (hp^.nodetype=addn) and is_constintnode(taddnode(hp^).right) then           begin             extraoffset:=tcgint(tordconstnode(taddnode(hp^).right).value);             replacenode(hp^,taddnode(hp^).left);           end         else if (hp^.nodetype=addn) and is_constintnode(taddnode(hp^).left) then           begin             extraoffset:=tcgint(tordconstnode(taddnode(hp^).left).value);             replacenode(hp^,taddnode(hp^).right);           end;         secondpass(left);         if not(left.location.loc in [LOC_CREGISTER,LOC_REGISTER,LOC_CREFERENCE,LOC_REFERENCE,LOC_CONSTANT]) then           hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,left.resultdef,left.resultdef,true);         case left.location.loc of            LOC_CREGISTER,            LOC_REGISTER:              begin                hlcg.maybe_change_load_node_reg(current_asmdata.CurrAsmList,left,true);              {$ifdef cpu_uses_separate_address_registers}                if getregtype(left.location.register)<>R_ADDRESSREGISTER then                  begin                    location.reference.base := cg.getaddressregister(current_asmdata.CurrAsmList);                    cg.a_load_reg_reg(current_asmdata.CurrAsmList,OS_ADDR,OS_ADDR,left.location.register,                      location.reference.base);                  end                else              {$endif}                  location.reference.base := left.location.register;              end;            LOC_CREFERENCE,            LOC_REFERENCE:              begin                 location.reference.base:=cg.getaddressregister(current_asmdata.CurrAsmList);                 hlcg.a_load_loc_reg(current_asmdata.CurrAsmList,left.resultdef,left.resultdef,left.location,location.reference.base);              end;            LOC_CONSTANT:              begin                location.reference.offset:=left.location.value;              end;            else              internalerror(200507031);         end;         location.reference.offset:=location.reference.offset+extraoffset;         if (cs_use_heaptrc in current_settings.globalswitches) and            (cs_checkpointer in current_settings.localswitches) and            not(cs_compilesystem in current_settings.moduleswitches) and            tpointerdef(left.resultdef).compatible_with_pointerdef_size(tpointerdef(voidpointertype)) and            not(nf_no_checkpointer in flags) and            { can be NR_NO in case of LOC_CONSTANT }            (location.reference.base<>NR_NO) then          begin            if not searchsym_in_named_module('HEAPTRC','CHECKPOINTER',sym,st) or               (sym.typ<>procsym) then              internalerror(2012010601);            pd:=tprocdef(tprocsym(sym).ProcdefList[0]);            paraloc1.init;            paramanager.getintparaloc(current_asmdata.CurrAsmList,pd,1,paraloc1);            hlcg.a_loadaddr_ref_cgpara(current_asmdata.CurrAsmList,left.resultdef,location.reference,paraloc1);            paramanager.freecgpara(current_asmdata.CurrAsmList,paraloc1);            paraloc1.done;            hlcg.allocallcpuregisters(current_asmdata.CurrAsmList);            hlcg.a_call_name(current_asmdata.CurrAsmList,pd,'FPC_CHECKPOINTER',[@paraloc1],nil,false);            hlcg.deallocallcpuregisters(current_asmdata.CurrAsmList);            include(current_settings.moduleswitches,cs_checkpointer_called);          end;      end;{*****************************************************************************                          TCGSUBSCRIPTNODE*****************************************************************************}    function tcgsubscriptnode.handle_platform_subscript: boolean;      begin        result:=false;      end;    procedure tcgsubscriptnode.pass_generate_code;      var        asmsym: tasmsymbol;        paraloc1 : tcgpara;        tmpref: treference;        sref: tsubsetreference;        awordoffset,        offsetcorrection : aint;        pd : tprocdef;        sym : tsym;        st : tsymtable;        hreg : TRegister;      begin         sym:=nil;         secondpass(left);         if codegenerror then           exit;         paraloc1.init;         { several object types must be dereferenced implicitly }         if is_implicit_pointer_object_type(left.resultdef) then           begin             if (not is_managed_type(left.resultdef)) or                (target_info.system in systems_garbage_collected_managed_types) then               begin                 { take care of the alignment of the fields }                 if not(left.resultdef is tabstractrecorddef) then                   Internalerror(2018021601);                 location_reset_ref(location,LOC_REFERENCE,def_cgsize(resultdef),newalignment(tabstractrecordsymtable(tabstractrecorddef(left.resultdef).symtable).recordalignment,vs.fieldoffset),[]);                 case left.location.loc of                    LOC_CREGISTER,                    LOC_REGISTER:                      begin                      {$ifdef cpu_uses_separate_address_registers}                        if getregtype(left.location.register)<>R_ADDRESSREGISTER then                          begin                            location.reference.base:=cg.getaddressregister(current_asmdata.CurrAsmList);                            hlcg.a_load_reg_reg(current_asmdata.CurrAsmList,left.resultdef,left.resultdef,                              left.location.register,location.reference.base);                          end                        else                      {$endif}                          hlcg.reference_reset_base(location.reference,left.resultdef,left.location.register,0,ctempposinvalid,location.reference.alignment,location.reference.volatility);                      end;                    LOC_CREFERENCE,                    LOC_REFERENCE,                    { tricky type casting of parameters can cause these locations, see tb0592.pp on x86_64-linux }                    LOC_SUBSETREG,                    LOC_CSUBSETREG,                    LOC_SUBSETREF,                    LOC_CSUBSETREF:                      begin                         hlcg.reference_reset_base(location.reference,left.resultdef,                           hlcg.getaddressregister(current_asmdata.CurrAsmList,left.resultdef),0,ctempposinvalid,location.reference.alignment,location.reference.volatility);                         hlcg.a_load_loc_reg(current_asmdata.CurrAsmList,left.resultdef,left.resultdef,left.location,location.reference.base);                      end;                    LOC_CONSTANT:                      begin                        { can happen with @classtype(pointerconst).field }                        location.reference.offset:=left.location.value;                      end;                    else                      internalerror(2009092401);                 end;                 { implicit deferencing }                 if (cs_use_heaptrc in current_settings.globalswitches) and                    (cs_checkpointer in current_settings.localswitches) and                    not(cs_compilesystem in current_settings.moduleswitches) then                  begin                    if not searchsym_in_named_module('HEAPTRC','CHECKPOINTER',sym,st) or                       (sym.typ<>procsym) then                      internalerror(2012010602);                    pd:=tprocdef(tprocsym(sym).ProcdefList[0]);                    paramanager.getintparaloc(current_asmdata.CurrAsmList,pd,1,paraloc1);                    hlcg.a_loadaddr_ref_cgpara(current_asmdata.CurrAsmList,left.resultdef,location.reference,paraloc1);                    paramanager.freecgpara(current_asmdata.CurrAsmList,paraloc1);                    hlcg.allocallcpuregisters(current_asmdata.CurrAsmList);                    hlcg.a_call_name(current_asmdata.CurrAsmList,pd,'FPC_CHECKPOINTER',[@paraloc1],nil,false);                    hlcg.deallocallcpuregisters(current_asmdata.CurrAsmList);                    system.include(current_settings.moduleswitches,cs_checkpointer_called);                  end;               end             else               { reference-counted implicit pointer object types don't have                 fields -> cannot be subscripted (calls are handled via call                 nodes) }               internalerror(2011011901);           end         else           begin             location_copy(location,left.location);             { some abi's require that functions return (some) records in }             { registers                                                  }             case location.loc of               LOC_REFERENCE,               LOC_CREFERENCE:                 ;               LOC_CONSTANT,               LOC_REGISTER,               LOC_CREGISTER,               { if a floating point value is casted into a record, it                 can happen that we get here an fpu or mm register }               LOC_MMREGISTER,               LOC_FPUREGISTER,               LOC_CMMREGISTER,               LOC_CFPUREGISTER:                 begin                   { in case the result is not something that can be put                     into an integer register (e.g.                     function_returning_record().non_regable_field, or                     a function returning a value > sizeof(intreg))                     -> force to memory                   }                   if not tstoreddef(left.resultdef).is_intregable or                      not tstoreddef(resultdef).is_intregable or                      { if the field spans multiple registers, we must force the record into                        memory as well }                      ((left.location.size in [OS_PAIR,OS_SPAIR]) and                       (vs.fieldoffset div sizeof(aword)<>(vs.fieldoffset+vs.getsize-1) div sizeof(aword))) or                      (location.loc in [LOC_FPUREGISTER,LOC_CFPUREGISTER,                        { actually, we should be able to "subscript" a constant, but this would require some code                          which enables dumping and reading constants from a temporary memory buffer. This                          must be done a CPU dependent way, so it is not easy and probably not worth the effort (FK)                        }                        LOC_CONSTANT]) then                     hlcg.location_force_mem(current_asmdata.CurrAsmList,location,left.resultdef)                   else                     begin                       if (location.loc in [LOC_MMREGISTER,LOC_CMMREGISTER]) then                         if (tcgsize2size[location.size]<=tcgsize2size[OS_INT]) then                           begin                             hreg:=cg.getintregister(current_asmdata.CurrAsmList,location.size);                             cg.a_loadmm_reg_intreg(current_asmdata.CurrAsmList,reg_cgsize(left.location.register),location.size,                               left.location.register,hreg,mms_movescalar);                             location_reset(left.location,LOC_REGISTER,int_cgsize(tcgsize2size[left.location.size]));                             left.location.register:=hreg;                             { copy again, we changed left.location }                             location_copy(location,left.location);                           end                         else                           hlcg.location_force_mem(current_asmdata.CurrAsmList,location,left.resultdef);                       if (left.location.loc = LOC_REGISTER) then                         location.loc := LOC_SUBSETREG                       else                         location.loc := LOC_CSUBSETREG;                       location.size:=def_cgsize(resultdef);                       offsetcorrection:=0;                       if (left.location.size in [OS_PAIR,OS_SPAIR]) then                         begin                           if not is_packed_record_or_object(left.resultdef) then                             awordoffset:=sizeof(aword)                           else                             awordoffset:=sizeof(aword)*8;                           if (vs.fieldoffset>=awordoffset) xor (target_info.endian=endian_big) then                             location.sreg.subsetreg := left.location.registerhi                           else                             location.sreg.subsetreg := left.location.register;                           if vs.fieldoffset>=awordoffset then                             offsetcorrection := sizeof(aword)*8;                           location.sreg.subsetregsize := OS_INT;                         end                       else                         begin                           location.sreg.subsetreg := left.location.register;                           location.sreg.subsetregsize := left.location.size;                         end;                       if not is_packed_record_or_object(left.resultdef) then                         begin                           if (target_info.endian = ENDIAN_BIG) then                             location.sreg.startbit := (tcgsize2size[location.sreg.subsetregsize] - tcgsize2size[location.size] - vs.fieldoffset) * 8+offsetcorrection                           else                             location.sreg.startbit := (vs.fieldoffset * 8)-offsetcorrection;                           location.sreg.bitlen := tcgsize2size[location.size] * 8;                         end                       else                         begin                           location.sreg.bitlen := resultdef.packedbitsize;                           if (target_info.endian = ENDIAN_BIG) then                             location.sreg.startbit := (tcgsize2size[location.sreg.subsetregsize]*8 - location.sreg.bitlen) - vs.fieldoffset+offsetcorrection                           else                             location.sreg.startbit := vs.fieldoffset-offsetcorrection;                         end;                     end;                 end;               LOC_SUBSETREG,               LOC_CSUBSETREG:                 begin                   location.size:=def_cgsize(resultdef);                   if not is_packed_record_or_object(left.resultdef) then                     begin                       if (target_info.endian = ENDIAN_BIG) then                         inc(location.sreg.startbit, (left.resultdef.size - tcgsize2size[location.size] - vs.fieldoffset) * 8)                       else                         inc(location.sreg.startbit, vs.fieldoffset * 8);                       location.sreg.bitlen := tcgsize2size[location.size] * 8;                     end                   else                     begin                       location.sreg.bitlen := resultdef.packedbitsize;                       if (target_info.endian = ENDIAN_BIG) then                         inc(location.sreg.startbit, left.location.sreg.bitlen - location.sreg.bitlen - vs.fieldoffset)                       else                         inc(location.sreg.startbit, vs.fieldoffset);                     end;                 end;               else                 internalerror(2006031901);             end;           end;         if is_objc_class_or_protocol(left.resultdef) and            (target_info.system in systems_objc_nfabi) then           begin             if (location.loc<>LOC_REFERENCE) or                (location.reference.index<>NR_NO) then               internalerror(2009092402);             { the actual field offset is stored in memory (to solve the               "fragile base class" problem: this way the layout of base               classes can be changed without breaking programs compiled against               earlier versions)             }             asmsym:=current_asmdata.RefAsmSymbol(vs.mangledname,AT_DATA);             reference_reset_symbol(tmpref,asmsym,0,voidpointertype.alignment,[]);             hlcg.g_ptrtypecast_ref(current_asmdata.CurrAsmList,left.resultdef,cpointerdef.getreusable(resultdef),location.reference);             location.reference.index:=hlcg.getintregister(current_asmdata.CurrAsmList,ptruinttype);             hlcg.a_load_ref_reg(current_asmdata.CurrAsmList,ptruinttype,ptruinttype,tmpref,location.reference.index);             { always packrecords C -> natural alignment }             location.reference.alignment:=vs.vardef.alignment;           end         else if handle_platform_subscript then           begin             { done }           end         else if (location.loc in [LOC_REFERENCE,LOC_CREFERENCE]) then           begin             if not is_packed_record_or_object(left.resultdef) then               begin                 inc(location.reference.offset,vs.fieldoffset);                 location.reference.alignment:=newalignment(location.reference.alignment,vs.fieldoffset);               end             else if (vs.fieldoffset mod 8 = 0) and                     (resultdef.packedbitsize mod 8 = 0) and                     { is different in case of e.g. packenum 2 and an enum }                     { which fits in 8 bits                                }                     (resultdef.size*8 = resultdef.packedbitsize) then               begin                 inc(location.reference.offset,vs.fieldoffset div 8);                 location.reference.alignment:=newalignment(location.reference.alignment,vs.fieldoffset div 8);               end             else               begin                 sref.ref:=location.reference;                 sref.ref.alignment:=1;                 sref.bitindexreg:=NR_NO;                 inc(sref.ref.offset,vs.fieldoffset div 8);                 sref.startbit:=vs.fieldoffset mod 8;                 sref.bitlen:=resultdef.packedbitsize;                 if (left.location.loc=LOC_REFERENCE) then                   location.loc:=LOC_SUBSETREF                 else                   location.loc:=LOC_CSUBSETREF;                 location.sref:=sref;               end;             { also update the size of the location }             location.size:=def_cgsize(resultdef);           end;         paraloc1.done;      end;{*****************************************************************************                            TCGVECNODE*****************************************************************************}     function tcgvecnode.get_mul_size : asizeint;       begin         if nf_memindex in flags then          get_mul_size:=1         else          begin            if (left.resultdef.typ=arraydef) then             if not is_packed_array(left.resultdef) then              get_mul_size:=tarraydef(left.resultdef).elesize             else              get_mul_size:=tarraydef(left.resultdef).elepackedbitsize            else             get_mul_size:=resultdef.size;          end       end;     { this routine must, like any other routine, not change the contents }     { of base/index registers of references, as these may be regvars.    }     { The register allocator can coalesce one LOC_REGISTER being moved   }     { into another (as their live ranges won't overlap), but not a       }     { LOC_CREGISTER moved into a LOC_(C)REGISTER most of the time (as    }     { the live range of the LOC_CREGISTER will most likely overlap the   }     { the live range of the target LOC_(C)REGISTER)                      }     { The passed register may be a LOC_CREGISTER as well.                }     procedure tcgvecnode.update_reference_reg_mul(maybe_const_reg: tregister; regsize: tdef; l: aint);       var         hreg: tregister;       begin         if l<>1 then           begin             hreg:=cg.getaddressregister(current_asmdata.CurrAsmList);             cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,OP_IMUL,OS_ADDR,l,maybe_const_reg,hreg);             maybe_const_reg:=hreg;           end;         if location.reference.base=NR_NO then           location.reference.base:=maybe_const_reg         else if location.reference.index=NR_NO then           location.reference.index:=maybe_const_reg         else          begin            hreg:=cg.getaddressregister(current_asmdata.CurrAsmList);            cg.a_loadaddr_ref_reg(current_asmdata.CurrAsmList,location.reference,hreg);            reference_reset_base(location.reference,hreg,0,location.reference.temppos,location.reference.alignment,location.reference.volatility);            { insert new index register }            location.reference.index:=maybe_const_reg;          end;          { update alignment }          if (location.reference.alignment=0) then            internalerror(2009020704);          location.reference.alignment:=newalignment(location.reference.alignment,l);       end;     { see remarks for tcgvecnode.update_reference_reg_mul above }     procedure tcgvecnode.update_reference_reg_packed(maybe_const_reg: tregister; regsize: tdef; l:aint);       var         sref: tsubsetreference;         offsetreg, hreg: tregister;         alignpower: aint;         temp : longint;       begin         { only orddefs are bitpacked. Even then we only need special code in }         { case the bitpacked *byte size* is not a power of two, otherwise    }         { everything can be handled using the the regular array code.        }         if ((l mod 8) = 0) and            (ispowerof2(l div 8,temp) or             not is_ordinal(resultdef){$ifndef cpu64bitalu}             or is_64bitint(resultdef){$endif not cpu64bitalu}             ) then           begin             update_reference_reg_mul(maybe_const_reg,regsize,l div 8);             exit;           end;         if (l > 8*sizeof(aint)) then           internalerror(200608051);         sref.ref := location.reference;         hreg := cg.getaddressregister(current_asmdata.CurrAsmList);         cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,OP_SUB,OS_ADDR,tarraydef(left.resultdef).lowrange,maybe_const_reg,hreg);         cg.a_op_const_reg(current_asmdata.CurrAsmList,OP_IMUL,OS_ADDR,l,hreg);         { keep alignment for index }         sref.ref.alignment := left.resultdef.alignment;         if not ispowerof2(packedbitsloadsize(l),temp) then           internalerror(2006081201);         alignpower:=temp;         offsetreg := cg.getaddressregister(current_asmdata.CurrAsmList);         cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,OP_SHR,OS_ADDR,3+alignpower,hreg,offsetreg);         cg.a_op_const_reg(current_asmdata.CurrAsmList,OP_SHL,OS_ADDR,alignpower,offsetreg);         if (sref.ref.base = NR_NO) then           sref.ref.base := offsetreg         else if (sref.ref.index = NR_NO) then           sref.ref.index := offsetreg         else           begin             cg.a_op_reg_reg(current_asmdata.CurrAsmList,OP_ADD,OS_ADDR,sref.ref.base,offsetreg);             sref.ref.base := offsetreg;           end;         { the if expression below is a constant evaluated at compile time, so disable the unreachable code           warning }{$push}{$warn 6018 off}         { we can reuse hreg only if OS_INT and OS_ADDR have the same size/type }         if OS_INT<>OS_ADDR then           begin             sref.bitindexreg := cg.getintregister(current_asmdata.CurrAsmList,OS_INT);             cg.a_load_reg_reg(current_asmdata.CurrAsmList,OS_ADDR,OS_INT,hreg,sref.bitindexreg);           end         else           sref.bitindexreg:=hreg;{$pop}         cg.a_op_const_reg(current_asmdata.CurrAsmList,OP_AND,OS_INT,(1 shl (3+alignpower))-1,sref.bitindexreg);         sref.startbit := 0;         sref.bitlen := resultdef.packedbitsize;         if (left.location.loc = LOC_REFERENCE) then           location.loc := LOC_SUBSETREF         else           location.loc := LOC_CSUBSETREF;         location.sref := sref;       end;     procedure tcgvecnode.update_reference_offset(var ref: treference; index, mulsize: ASizeInt);       begin         inc(ref.offset,index*mulsize);       end;     procedure tcgvecnode.second_wideansistring;       begin       end;     procedure tcgvecnode.second_dynamicarray;       begin       end;     function tcgvecnode.valid_index_size(size: tcgsize): boolean;       begin         result:=           tcgsize2signed[size]=tcgsize2signed[OS_ADDR];       end;     procedure tcgvecnode.rangecheck_array;       var         paraloc1,paraloc2 : tcgpara;         pd : tprocdef;       begin         { omit range checking when this is an array access to a pointer which has been           typecasted from an array }         if (ado_isconvertedpointer in tarraydef(left.resultdef).arrayoptions) then           exit;         paraloc1.init;         paraloc2.init;         if is_dynamic_array(left.resultdef) then            begin               pd:=search_system_proc('fpc_dynarray_rangecheck');               paramanager.getintparaloc(current_asmdata.CurrAsmList,pd,1,paraloc1);               paramanager.getintparaloc(current_asmdata.CurrAsmList,pd,2,paraloc2);               if pd.is_pushleftright then                 begin                   hlcg.a_load_loc_cgpara(current_asmdata.CurrAsmList,left.resultdef,left.location,paraloc1);                   hlcg.a_load_loc_cgpara(current_asmdata.CurrAsmList,right.resultdef,right.location,paraloc2);                 end               else                 begin                   hlcg.a_load_loc_cgpara(current_asmdata.CurrAsmList,right.resultdef,right.location,paraloc2);                   hlcg.a_load_loc_cgpara(current_asmdata.CurrAsmList,left.resultdef,left.location,paraloc1);                 end;               paramanager.freecgpara(current_asmdata.CurrAsmList,paraloc1);               paramanager.freecgpara(current_asmdata.CurrAsmList,paraloc2);               hlcg.g_call_system_proc(current_asmdata.CurrAsmList,pd,[@paraloc1,@paraloc2],nil).resetiftemp;            end;         { for regular arrays, we don't have to do anything because the index has been           type converted to the index type, which already inserted a range check if           necessary }         paraloc1.done;         paraloc2.done;       end;    procedure tcgvecnode.rangecheck_string;      var        paraloc1,        paraloc2: tcgpara;        helpername: TIDString;        pd: tprocdef;      begin        paraloc1.init;        paraloc2.init;        case tstringdef(left.resultdef).stringtype of          { it's the same for ansi- and wide strings }          st_unicodestring,          st_widestring,          st_ansistring:            begin              helpername:='fpc_'+tstringdef(left.resultdef).stringtypname+'_rangecheck';              pd:=search_system_proc(helpername);              paramanager.getintparaloc(current_asmdata.CurrAsmList,pd,1,paraloc1);              paramanager.getintparaloc(current_asmdata.CurrAsmList,pd,2,paraloc2);              if pd.is_pushleftright then                begin                  hlcg.a_load_loc_cgpara(current_asmdata.CurrAsmList,left.resultdef,left.location,paraloc1);                  hlcg.a_load_loc_cgpara(current_asmdata.CurrAsmList,right.resultdef,right.location,paraloc2);                end              else                begin                  hlcg.a_load_loc_cgpara(current_asmdata.CurrAsmList,right.resultdef,right.location,paraloc2);                  hlcg.a_load_loc_cgpara(current_asmdata.CurrAsmList,left.resultdef,left.location,paraloc1);                end;              paramanager.freecgpara(current_asmdata.CurrAsmList,paraloc1);              paramanager.freecgpara(current_asmdata.CurrAsmList,paraloc2);              hlcg.g_call_system_proc(current_asmdata.CurrAsmList,pd,[@paraloc1,@paraloc2],nil).resetiftemp;            end;          st_shortstring:            begin              {!!!!!!!!!!!!!!!!!}              { if this one is implemented making use of the high parameter for openshortstrings, update ncgutils.do_get_used_regvars() too (JM) }            end;          st_longstring:            begin              {!!!!!!!!!!!!!!!!!}            end;        end;        paraloc1.done;        paraloc2.done;      end;    procedure tcgvecnode.pass_generate_code;      var         offsetdec,         extraoffset : ASizeInt;         rightp      : pnode;         newsize  : tcgsize;         mulsize,         bytemulsize : ASizeInt;         alignpow : aint;         paraloc1,         paraloc2 : tcgpara;         subsetref : tsubsetreference;         temp : longint;         indexdef : tdef;      begin         paraloc1.init;         paraloc2.init;         mulsize:=get_mul_size;         if not is_packed_array(left.resultdef) then           bytemulsize:=mulsize         else           bytemulsize:=mulsize div 8;         newsize:=def_cgsize(resultdef);         secondpass(left);         if left.location.loc in [LOC_CREFERENCE,LOC_REFERENCE] then           location_reset_ref(location,left.location.loc,newsize,left.location.reference.alignment,left.location.reference.volatility)         else           location_reset_ref(location,LOC_REFERENCE,newsize,resultdef.alignment,[]);         { an ansistring needs to be dereferenced }         if is_ansistring(left.resultdef) or            is_wide_or_unicode_string(left.resultdef) then           begin              if nf_callunique in flags then                internalerror(200304236);              {DM!!!!!}              case left.location.loc of                LOC_REGISTER,                LOC_CREGISTER :                  begin                    hlcg.reference_reset_base(location.reference,left.resultdef,left.location.register,0,ctempposinvalid,location.reference.alignment,[]);                  end;                LOC_CREFERENCE,                LOC_REFERENCE :                  begin                    hlcg.reference_reset_base(location.reference,left.resultdef,hlcg.getaddressregister(current_asmdata.CurrAsmList,left.resultdef),0,ctempposinvalid,location.reference.alignment,[]);                    hlcg.a_load_ref_reg(current_asmdata.CurrAsmList,left.resultdef,left.resultdef,left.location.reference,location.reference.base);                  end;                LOC_CONSTANT:                  begin                    hlcg.reference_reset_base(location.reference,left.resultdef,NR_NO,left.location.value,ctempposinvalid,location.reference.alignment,[]);                  end;                else                  internalerror(2002032218);              end;              if is_ansistring(left.resultdef) then                offsetdec:=1              else                offsetdec:=2;              location.reference.alignment:=offsetdec;              location.reference.volatility:=[];              { in ansistrings/widestrings S[1] is p<w>char(S)[0] }              if not(cs_zerobasedstrings in current_settings.localswitches) then                update_reference_offset(location.reference,-1,offsetdec);           end         else if is_dynamic_array(left.resultdef) then           begin              case left.location.loc of                LOC_REGISTER,                LOC_CREGISTER :                  hlcg.reference_reset_base(location.reference,left.resultdef,left.location.register,0,ctempposinvalid,location.reference.alignment,[]);                LOC_REFERENCE,                LOC_CREFERENCE :                  begin                     hlcg.reference_reset_base(location.reference,left.resultdef,hlcg.getaddressregister(current_asmdata.CurrAsmList,left.resultdef),0,ctempposinvalid,location.reference.alignment,[]);                     hlcg.a_load_ref_reg(current_asmdata.CurrAsmList,left.resultdef,left.resultdef,                      left.location.reference,location.reference.base);                  end;                else                  internalerror(2002032219);              end;              { a dynarray points to the start of a memory block, which                we assume to be always aligned to a multiple of the                pointer size              }              location.reference.alignment:=voidpointertype.size;              location.reference.volatility:=[];           end         else           begin              { may happen in case of function results }              case left.location.loc of                LOC_CSUBSETREG,                LOC_CREGISTER,                LOC_CMMREGISTER,                LOC_SUBSETREG,                LOC_REGISTER,                LOC_MMREGISTER:                  hlcg.location_force_mem(current_asmdata.CurrAsmList,left.location,left.resultdef);              end;             location_copy(location,left.location);           end;         { location must be memory }         if not(location.loc in [LOC_REFERENCE,LOC_CREFERENCE]) then           internalerror(200411013);         { offset can only differ from 0 if arraydef }         if (left.resultdef.typ=arraydef) and            not(is_dynamic_array(left.resultdef)) and            (not(is_packed_array(left.resultdef)) or             ((mulsize mod 8 = 0) and              ispowerof2(mulsize div 8,temp)) or              { only orddefs are bitpacked }              not is_ordinal(resultdef){$ifndef cpu64bitalu}              or is_64bitint(resultdef){$endif not cpu64bitalu}              ) then           update_reference_offset(location.reference,-tarraydef(left.resultdef).lowrange,bytemulsize);         if right.nodetype=ordconstn then           begin              { offset can only differ from 0 if arraydef }              if cs_check_range in current_settings.localswitches then                begin                  secondpass(right);                  case left.resultdef.typ of                    arraydef :                      rangecheck_array;                    stringdef :                      rangecheck_string;                  end;                end;              if not(is_packed_array(left.resultdef)) or                 ((mulsize mod 8 = 0) and                  (ispowerof2(mulsize div 8,temp) or                   { only orddefs are bitpacked }                   not is_ordinal(resultdef))) then                begin                  extraoffset:=tordconstnode(right).value.svalue;                  update_reference_offset(location.reference,extraoffset,bytemulsize);                  { adjust alignment after this change }                  location.reference.alignment:=newalignment(location.reference.alignment,extraoffset*bytemulsize);                end              else                begin                  subsetref.ref := location.reference;                  subsetref.ref.alignment := left.resultdef.alignment;                  if not ispowerof2(packedbitsloadsize(resultdef.packedbitsize),temp) then                    internalerror(2006081212);                  alignpow:=temp;                  update_reference_offset(subsetref.ref,(mulsize * (tordconstnode(right).value.svalue-tarraydef(left.resultdef).lowrange)) shr (3+alignpow),1 shl alignpow);                  subsetref.bitindexreg := NR_NO;                  subsetref.startbit := (mulsize * (tordconstnode(right).value.svalue-tarraydef(left.resultdef).lowrange)) and ((1 shl (3+alignpow))-1);                  subsetref.bitlen := resultdef.packedbitsize;                  if (left.location.loc = LOC_REFERENCE) then                    location.loc := LOC_SUBSETREF                  else                    location.loc := LOC_CSUBSETREF;                  location.sref := subsetref;                end;           end         else         { not nodetype=ordconstn }           begin              if (cs_opt_level1 in current_settings.optimizerswitches) and                 { if we do range checking, we don't }                 { need that fancy code (it would be }                 { buggy)                            }                 not(cs_check_range in current_settings.localswitches) and                 (left.resultdef.typ=arraydef) and                 not is_packed_array(left.resultdef) then                begin                   extraoffset:=0;                   rightp:=actualtargetnode(@right);                   if rightp^.nodetype=addn then                     begin                        if taddnode(rightp^).right.nodetype=ordconstn then                          begin                            extraoffset:=tordconstnode(taddnode(rightp^).right).value.svalue;                            replacenode(rightp^,taddnode(rightp^).left);                          end                        else if taddnode(rightp^).left.nodetype=ordconstn then                          begin                            extraoffset:=tordconstnode(taddnode(rightp^).left).value.svalue;                            replacenode(rightp^,taddnode(rightp^).right);                          end;                     end                   else if rightp^.nodetype=subn then                     begin                        if taddnode(rightp^).right.nodetype=ordconstn then                          begin                            extraoffset:=-tordconstnode(taddnode(rightp^).right).value.svalue;                            replacenode(rightp^,taddnode(rightp^).left);                          end;                     end;                   update_reference_offset(location.reference,extraoffset,mulsize);                end;              { calculate from left to right }              if not(location.loc in [LOC_CREFERENCE,LOC_REFERENCE]) then                internalerror(200304237);              secondpass(right);              if (right.expectloc=LOC_JUMP)<>                 (right.location.loc=LOC_JUMP) then                internalerror(2006010801);              { if mulsize = 1, we won't have to modify the index }              if not(right.location.loc in [LOC_CREGISTER,LOC_REGISTER]) or                 not valid_index_size(right.location.size) then                begin                  hlcg.location_force_reg(current_asmdata.CurrAsmList,right.location,right.resultdef,sizeuinttype,true);                  indexdef:=sizeuinttype                end              else                indexdef:=right.resultdef;            { produce possible range check code: }              if cs_check_range in current_settings.localswitches then               begin                 if left.resultdef.typ=arraydef then                   rangecheck_array                 else if (left.resultdef.typ=stringdef) then                   rangecheck_string;               end;              { insert the register and the multiplication factor in the                reference }              if not is_packed_array(left.resultdef) then                update_reference_reg_mul(right.location.register,indexdef,mulsize)              else                update_reference_reg_packed(right.location.register,indexdef,mulsize);           end;        location.size:=newsize;        paraloc1.done;        paraloc2.done;      end;begin   cloadvmtaddrnode:=tcgloadvmtaddrnode;   cloadparentfpnode:=tcgloadparentfpnode;   caddrnode:=tcgaddrnode;   cderefnode:=tcgderefnode;   csubscriptnode:=tcgsubscriptnode;   cvecnode:=tcgvecnode;end.
 |