| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022 | {    Copyright (c) 2000-2002 by Florian Klaempfl    Type checking and register allocation for 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 nmem;{$i fpcdefs.inc}interface    uses       node,       symdef,symsym,symtable,symtype;    type       tloadvmtaddrnode = class(tunarynode)          constructor create(l : tnode);virtual;          function pass_1 : tnode;override;          function pass_typecheck:tnode;override;       end;       tloadvmtaddrnodeclass = class of tloadvmtaddrnode;       tloadparentfpnode = class(tunarynode)          parentpd : tprocdef;          parentpdderef : tderef;          constructor create(pd:tprocdef);virtual;          constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderefimpl;override;          procedure derefimpl;override;          function pass_1 : tnode;override;          function pass_typecheck:tnode;override;          function dogetcopy : tnode;override;       end;       tloadparentfpnodeclass = class of tloadparentfpnode;       taddrnode = class(tunarynode)          getprocvardef : tprocvardef;          getprocvardefderef : tderef;          constructor create(l : tnode);virtual;          constructor create_internal(l : tnode); virtual;          constructor create_internal_nomark(l : tnode); virtual;          constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure mark_write;override;          procedure buildderefimpl;override;          procedure derefimpl;override;          function dogetcopy : tnode;override;          function pass_1 : tnode;override;          function pass_typecheck:tnode;override;         private          mark_read_written: boolean;       end;       taddrnodeclass = class of taddrnode;       tderefnode = class(tunarynode)          constructor create(l : tnode);virtual;          function pass_1 : tnode;override;          function pass_typecheck:tnode;override;          procedure mark_write;override;       end;       tderefnodeclass = class of tderefnode;       tsubscriptnode = class(tunarynode)          vs : tfieldvarsym;          vsderef : tderef;          constructor create(varsym : tsym;l : tnode);virtual;          constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderefimpl;override;          procedure derefimpl;override;          function dogetcopy : tnode;override;          function pass_1 : tnode;override;          function docompare(p: tnode): boolean; override;          function pass_typecheck:tnode;override;          procedure mark_write;override;       end;       tsubscriptnodeclass = class of tsubscriptnode;       tvecnode = class(tbinarynode)          constructor create(l,r : tnode);virtual;          function pass_1 : tnode;override;          function pass_typecheck:tnode;override;          procedure mark_write;override;       end;       tvecnodeclass = class of tvecnode;       twithnode = class(tunarynode)          constructor create(l:tnode);          destructor destroy;override;          constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function dogetcopy : tnode;override;          function pass_1 : tnode;override;          function docompare(p: tnode): boolean; override;          function pass_typecheck:tnode;override;       end;       twithnodeclass = class of twithnode;    var       cloadvmtaddrnode : tloadvmtaddrnodeclass;       cloadparentfpnode : tloadparentfpnodeclass;       caddrnode : taddrnodeclass;       cderefnode : tderefnodeclass;       csubscriptnode : tsubscriptnodeclass;       cvecnode : tvecnodeclass;       cwithnode : twithnodeclass;    function is_big_untyped_addrnode(p: tnode): boolean;implementation    uses      globtype,systems,      cutils,verbose,globals,      symconst,symbase,defutil,defcmp,      nbas,nutils,      htypechk,pass_1,ncal,nld,ncon,ncnv,cgbase,procinfo      ;{*****************************************************************************                            TLOADVMTADDRNODE*****************************************************************************}    constructor tloadvmtaddrnode.create(l : tnode);      begin         inherited create(loadvmtaddrn,l);      end;    function tloadvmtaddrnode.pass_typecheck:tnode;      begin        result:=nil;        typecheckpass(left);        if codegenerror then         exit;        case left.resultdef.typ of          classrefdef :            resultdef:=left.resultdef;          objectdef :            resultdef:=tclassrefdef.create(left.resultdef);          else            Message(parser_e_pointer_to_class_expected);        end;      end;    function tloadvmtaddrnode.pass_1 : tnode;      begin         result:=nil;         expectloc:=LOC_REGISTER;         if left.nodetype<>typen then           begin             firstpass(left);             registersint:=left.registersint;           end;         if registersint<1 then           registersint:=1;      end;{*****************************************************************************                        TLOADPARENTFPNODE*****************************************************************************}    constructor tloadparentfpnode.create(pd:tprocdef);      begin        inherited create(loadparentfpn,nil);        if not assigned(pd) then          internalerror(200309288);        if (pd.parast.symtablelevel>current_procinfo.procdef.parast.symtablelevel) then          internalerror(200309284);        parentpd:=pd;      end;    constructor tloadparentfpnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);      begin        inherited ppuload(t,ppufile);        ppufile.getderef(parentpdderef);      end;    procedure tloadparentfpnode.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);        ppufile.putderef(parentpdderef);      end;    procedure tloadparentfpnode.buildderefimpl;      begin        inherited buildderefimpl;        parentpdderef.build(parentpd);      end;    procedure tloadparentfpnode.derefimpl;      begin        inherited derefimpl;        parentpd:=tprocdef(parentpdderef.resolve);      end;    function tloadparentfpnode.dogetcopy : tnode;      var         p : tloadparentfpnode;      begin         p:=tloadparentfpnode(inherited dogetcopy);         p.parentpd:=parentpd;         dogetcopy:=p;      end;    function tloadparentfpnode.pass_typecheck:tnode;{$ifdef dummy}      var        currpi : tprocinfo;        hsym   : tparavarsym;{$endif dummy}      begin        result:=nil;        resultdef:=voidpointertype;{$ifdef dummy}        { currently parentfps are never loaded in registers (FK) }        if (current_procinfo.procdef.parast.symtablelevel<>parentpd.parast.symtablelevel) then          begin            currpi:=current_procinfo;            { walk parents }            while (currpi.procdef.owner.symtablelevel>parentpd.parast.symtablelevel) do              begin                currpi:=currpi.parent;                if not assigned(currpi) then                  internalerror(2005040602);                hsym:=tparavarsym(currpi.procdef.parast.Find('parentfp'));                if not assigned(hsym) then                  internalerror(2005040601);                hsym.varregable:=vr_none;              end;          end;{$endif dummy}      end;    function tloadparentfpnode.pass_1 : tnode;      begin        result:=nil;        expectloc:=LOC_REGISTER;        registersint:=1;      end;{*****************************************************************************                             TADDRNODE*****************************************************************************}    constructor taddrnode.create(l : tnode);      begin         inherited create(addrn,l);         getprocvardef:=nil;         mark_read_written := true;      end;    constructor taddrnode.create_internal(l : tnode);      begin        self.create(l);        include(flags,nf_internal);      end;    constructor taddrnode.create_internal_nomark(l : tnode);      begin        self.create_internal(l);        mark_read_written := false;      end;    constructor taddrnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);      begin        inherited ppuload(t,ppufile);        ppufile.getderef(getprocvardefderef);      end;    procedure taddrnode.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);        ppufile.putderef(getprocvardefderef);      end;    procedure Taddrnode.mark_write;    begin      {@procvar:=nil is legal in Delphi mode.}      left.mark_write;    end;    procedure taddrnode.buildderefimpl;      begin        inherited buildderefimpl;        getprocvardefderef.build(getprocvardef);      end;    procedure taddrnode.derefimpl;      begin        inherited derefimpl;        getprocvardef:=tprocvardef(getprocvardefderef.resolve);      end;    function taddrnode.dogetcopy : tnode;      var         p : taddrnode;      begin         p:=taddrnode(inherited dogetcopy);         p.getprocvardef:=getprocvardef;         dogetcopy:=p;      end;    function taddrnode.pass_typecheck:tnode;      var         hp  : tnode;         hsym : tfieldvarsym;         isprocvar : boolean;      begin        result:=nil;        typecheckpass(left);        if codegenerror then         exit;        make_not_regable(left,[ra_addr_regable,ra_addr_taken]);        { don't allow constants, for internal use we also          allow taking the address of strings }        if is_constnode(left) and           not(               (nf_internal in flags) and               (left.nodetype in [stringconstn])              ) then         begin           current_filepos:=left.fileinfo;           CGMessage(type_e_no_addr_of_constant);           exit;         end;        { Handle @proc special, also @procvar in tp-mode needs          special handling }        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            isprocvar:=(left.resultdef.typ=procvardef);            if not isprocvar then              begin                left:=ctypeconvnode.create_proc_to_procvar(left);                typecheckpass(left);              end;            { In tp procvar mode the result is always a voidpointer. Insert              a typeconversion to voidpointer. For methodpointers we need              to load the proc field }            if (m_tp_procvar in current_settings.modeswitches) or               (m_mac_procvar in current_settings.modeswitches) then              begin                if tabstractprocdef(left.resultdef).is_addressonly then                  begin                    result:=ctypeconvnode.create_internal(left,voidpointertype);                    include(result.flags,nf_load_procvar);                    left:=nil;                  end                else                  begin                    { For procvars we need to return the proc field of the                      methodpointer }                    if isprocvar then                      begin                        { find proc field in methodpointer record }                        hsym:=tfieldvarsym(trecorddef(methodpointertype).symtable.Find('proc'));                        if not assigned(hsym) then                          internalerror(200412041);                        { Load tmehodpointer(left).proc }                        result:=csubscriptnode.create(                                     hsym,                                     ctypeconvnode.create_internal(left,methodpointertype));                        left:=nil;                      end                    else                      CGMessage(type_e_variable_id_expected);                  end;              end            else              begin                { Return the typeconvn only }                result:=left;                left:=nil;              end;          end        else          begin            { what are we getting the address from an absolute sym? }            hp:=left;            while assigned(hp) and (hp.nodetype in [typeconvn,vecn,derefn,subscriptn]) do              hp:=tunarynode(hp).left;            if not assigned(hp) then              internalerror(200412042);{$ifdef i386}            if (hp.nodetype=loadn) and               ((tloadnode(hp).symtableentry.typ=absolutevarsym) and               tabsolutevarsym(tloadnode(hp).symtableentry).absseg) then              begin                if not(nf_typedaddr in flags) then                  resultdef:=voidfarpointertype                else                  resultdef:=tpointerdef.createfar(left.resultdef);              end            else{$endif i386}              if (nf_internal in flags) or                 valid_for_addr(left,true) then                begin                  if not(nf_typedaddr in flags) then                    resultdef:=voidpointertype                  else                    resultdef:=tpointerdef.create(left.resultdef);                end            else              CGMessage(type_e_variable_id_expected);          end;        if (mark_read_written) then          begin            { this is like the function addr }            inc(parsing_para_level);            { This is actually only "read", but treat it nevertheless as  }            { modified due to the possible use of pointers                }            { To avoid false positives regarding "uninitialised"          }            { warnings when using arrays, perform it in two steps         }            set_varstate(left,vs_written,[]);            { vsf_must_be_valid so it doesn't get changed into }            { vsf_referred_not_inited                          }            set_varstate(left,vs_read,[vsf_must_be_valid]);            dec(parsing_para_level);          end;      end;    function taddrnode.pass_1 : tnode;      begin         result:=nil;         firstpass(left);         if codegenerror then          exit;         registersint:=left.registersint;         registersfpu:=left.registersfpu;{$ifdef SUPPORT_MMX}         registersmmx:=left.registersmmx;{$endif SUPPORT_MMX}         if registersint<1 then           registersint:=1;         { is this right for object of methods ?? }         expectloc:=LOC_REGISTER;      end;{*****************************************************************************                             TDEREFNODE*****************************************************************************}    constructor tderefnode.create(l : tnode);      begin         inherited create(derefn,l);      end;    function tderefnode.pass_typecheck:tnode;      begin         result:=nil;         typecheckpass(left);         set_varstate(left,vs_read,[vsf_must_be_valid]);         if codegenerror then          exit;         { tp procvar support }         maybe_call_procvar(left,true);         if left.resultdef.typ=pointerdef then          resultdef:=tpointerdef(left.resultdef).pointeddef         else          CGMessage(parser_e_invalid_qualifier);      end;    procedure Tderefnode.mark_write;    begin      include(flags,nf_write);    end;    function tderefnode.pass_1 : tnode;      begin         result:=nil;         firstpass(left);         if codegenerror then          exit;         registersint:=max(left.registersint,1);         registersfpu:=left.registersfpu;{$ifdef SUPPORT_MMX}         registersmmx:=left.registersmmx;{$endif SUPPORT_MMX}         expectloc:=LOC_REFERENCE;      end;{*****************************************************************************                            TSUBSCRIPTNODE*****************************************************************************}    constructor tsubscriptnode.create(varsym : tsym;l : tnode);      begin         inherited create(subscriptn,l);         { vs should be changed to tsym! }         vs:=tfieldvarsym(varsym);      end;    constructor tsubscriptnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);      begin        inherited ppuload(t,ppufile);        ppufile.getderef(vsderef);      end;    procedure tsubscriptnode.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);        ppufile.putderef(vsderef);      end;    procedure tsubscriptnode.buildderefimpl;      begin        inherited buildderefimpl;        vsderef.build(vs);      end;    procedure tsubscriptnode.derefimpl;      begin        inherited derefimpl;        vs:=tfieldvarsym(vsderef.resolve);      end;    function tsubscriptnode.dogetcopy : tnode;      var         p : tsubscriptnode;      begin         p:=tsubscriptnode(inherited dogetcopy);         p.vs:=vs;         dogetcopy:=p;      end;    function tsubscriptnode.pass_typecheck:tnode;      begin        result:=nil;        typecheckpass(left);        { tp procvar support }        maybe_call_procvar(left,true);        resultdef:=vs.vardef;        // don't put records from which we load fields which aren't regable in integer registers        if (left.resultdef.typ = recorddef) and           not(tstoreddef(resultdef).is_intregable) then          make_not_regable(left,[ra_addr_regable]);      end;    procedure Tsubscriptnode.mark_write;    begin      include(flags,nf_write);    end;    function tsubscriptnode.pass_1 : tnode;      begin         result:=nil;         firstpass(left);         if codegenerror then          exit;         registersint:=left.registersint;         registersfpu:=left.registersfpu;{$ifdef SUPPORT_MMX}         registersmmx:=left.registersmmx;{$endif SUPPORT_MMX}         { classes must be dereferenced implicit }         if is_class_or_interface(left.resultdef) then           begin              if registersint=0 then                registersint:=1;              expectloc:=LOC_REFERENCE;           end         else           begin             case left.expectloc of               LOC_REGISTER,               LOC_SUBSETREG:                 // can happen for function results on win32 and darwin/x86                 if (left.resultdef.size > sizeof(aint)) then                   expectloc:=LOC_REFERENCE                 else                   expectloc:=LOC_SUBSETREG;               LOC_CREGISTER,               LOC_CSUBSETREG:                 expectloc:=LOC_CSUBSETREG;               LOC_REFERENCE,               LOC_CREFERENCE:                 expectloc:=left.expectloc;               else internalerror(20060521);              end;           end;      end;    function tsubscriptnode.docompare(p: tnode): boolean;      begin        docompare :=          inherited docompare(p) and          (vs = tsubscriptnode(p).vs);      end;{*****************************************************************************                               TVECNODE*****************************************************************************}    constructor tvecnode.create(l,r : tnode);      begin         inherited create(vecn,l,r);      end;    function tvecnode.pass_typecheck:tnode;      var         hightree: tnode;         htype,elementdef : tdef;         valid : boolean;      begin         result:=nil;         typecheckpass(left);         typecheckpass(right);         { implicitly convert stringconstant to stringdef,           see tbs/tb0476.pp for a test }         if (left.nodetype=stringconstn) and            (tstringconstnode(left).cst_type=cst_conststring) then           begin             if tstringconstnode(left).len>255 then               inserttypeconv(left,cansistringtype)             else               inserttypeconv(left,cshortstringtype);           end;         { In p[1] p is always valid, it is not possible to           declared a shortstring or normal array that has           undefined number of elements. Dynamic array and           ansi/widestring needs to be valid }         valid:=is_dynamic_array(left.resultdef) or                is_ansistring(left.resultdef) or                is_widestring(left.resultdef) or                { implicit pointer dereference -> pointer is read }                (left.resultdef.typ = pointerdef);         if valid then           set_varstate(left,vs_read,[vsf_must_be_valid]);{         A vecn is, just like a loadn, always part of an expression with its         own read/write and must_be_valid semantics. Therefore we don't have         to do anything else here, just like for loadn's}         set_varstate(right,vs_read,[vsf_must_be_valid]);         if codegenerror then          exit;         { maybe type conversion for the index value, but           do not convert enums,booleans,char           and do not convert range nodes }         if (right.nodetype<>rangen) and (is_integer(right.resultdef) or (left.resultdef.typ<>arraydef)) then           case left.resultdef.typ of             arraydef:               if ado_isvariant in Tarraydef(left.resultdef).arrayoptions then                 {Variant arrays are a special array, can have negative indexes and would therefore                  need s32bit. However, they should not appear in a vecn, as they are handled in                  handle_variantarray in pexpr.pas. Therefore, encountering a variant array is an                  internal error... }                 internalerror(200707031)               else if is_special_array(left.resultdef) then                 {Arrays without a high bound (dynamic arrays, open arrays) are zero based,                  convert indexes into these arrays to aword.}                 inserttypeconv(right,uinttype)               else                 {Convert array indexes to low_bound..high_bound.}                 inserttypeconv(right,Torddef.create(Torddef(sinttype).ordtype,                                                     int64(Tarraydef(left.resultdef).lowrange),                                                     int64(Tarraydef(left.resultdef).highrange)                                                    ));             stringdef:               if is_open_string(left.resultdef) then                 inserttypeconv(right,u8inttype)               else if is_shortstring(left.resultdef) then                 {Convert shortstring indexes to 0..length.}                 inserttypeconv(right,Torddef.create(u8bit,0,int64(Tstringdef(left.resultdef).len)))               else                 {Convert indexes into dynamically allocated strings to aword.}                 inserttypeconv(right,uinttype);             else               {Others, i.e. pointer indexes to aint.}               inserttypeconv(right,sinttype);           end;         case left.resultdef.typ of           arraydef :             begin               { check type of the index value }               if (compare_defs(right.resultdef,tarraydef(left.resultdef).rangedef,right.nodetype)=te_incompatible) then                 IncompatibleTypes(right.resultdef,tarraydef(left.resultdef).rangedef);               if right.nodetype=rangen then                 resultdef:=left.resultdef               else                 resultdef:=Tarraydef(left.resultdef).elementdef;               { if we are range checking an open array or array of const, we }               { need to load the high parameter. If the current procedure is }               { nested inside the procedure to which the open array/of const }               { was passed, then the high parameter must not be a regvar.    }               { So create a loadnode for the high parameter here and         }               { typecheck it, then the loadnode will make the high parameter }               { not regable. Otherwise this would only happen inside pass_2, }               { which is too late since by then the regvars are already      }               { assigned (pass_1 is also already too late, because then the  }               { regvars of the parent are also already assigned).            }               { webtbs/tw8975                                                }               if (cs_check_range in current_settings.localswitches) and                  (is_open_array(left.resultdef) or                   is_array_of_const(left.resultdef)) and                  { cdecl functions don't have high() so we can not check the range }                  { (can't use current_procdef, since it may be a nested procedure) }                  not(tprocdef(tparasymtable(tparavarsym(tloadnode(left).symtableentry).owner).defowner).proccalloption in [pocall_cdecl,pocall_cppdecl]) then                   begin                     { load_high_value_node already typechecks }                     hightree:=load_high_value_node(tparavarsym(tloadnode(left).symtableentry));                     hightree.free;                   end;                         end;           pointerdef :             begin               { are we accessing a pointer[], then convert the pointer to                 an array first, in FPC this is allowed for all pointers                 (except voidpointer) in delphi/tp7 it's only allowed for pchars. }               if not is_voidpointer(left.resultdef) and                  (                   (m_fpc in current_settings.modeswitches) or                   is_pchar(left.resultdef) or                   is_pwidechar(left.resultdef)                  ) then                begin                  { convert pointer to array }                  htype:=tarraydef.create_from_pointer(tpointerdef(left.resultdef).pointeddef);                  inserttypeconv(left,htype);                  if right.nodetype=rangen then                    resultdef:=htype                  else                    resultdef:=tarraydef(htype).elementdef;                end               else                CGMessage(type_e_array_required);             end;           stringdef :             begin                case tstringdef(left.resultdef).stringtype of                  st_widestring :                    elementdef:=cwidechartype;                  st_ansistring :                    elementdef:=cchartype;                  st_longstring :                    elementdef:=cchartype;                  st_shortstring :                    elementdef:=cchartype;                end;                if right.nodetype=rangen then                  begin                    htype:=Tarraydef.create_from_pointer(elementdef);                    resultdef:=htype;                  end                else                 begin                   { indexed access to 0 element is only allowed for shortstrings }                   if (right.nodetype=ordconstn) and                      (tordconstnode(right).value=0) and                      not is_shortstring(left.resultdef) then                     CGMessage(cg_e_can_access_element_zero);                   resultdef:=elementdef;                 end;             end;           variantdef :             resultdef:=cvarianttype;           else             CGMessage(type_e_array_required);        end;      end;    procedure Tvecnode.mark_write;    begin      include(flags,nf_write);    end;    function tvecnode.pass_1 : tnode;      begin         result:=nil;         firstpass(left);         firstpass(right);         if codegenerror then           exit;         if (nf_callunique in flags) and            (is_ansistring(left.resultdef) or             (is_widestring(left.resultdef) and not(tf_winlikewidestring in target_info.flags))) then           begin             left := ctypeconvnode.create_internal(ccallnode.createintern('fpc_'+tstringdef(left.resultdef).stringtypname+'_unique',               ccallparanode.create(                 ctypeconvnode.create_internal(left,voidpointertype),nil)),               left.resultdef);             firstpass(left);             { double resultdef passes somwhere else may cause this to be }             { reset though :/                                             }             exclude(flags,nf_callunique);           end         else if is_widestring(left.resultdef) and (tf_winlikewidestring in target_info.flags) then           exclude(flags,nf_callunique);         { the register calculation is easy if a const index is used }         if right.nodetype=ordconstn then           begin              registersint:=left.registersint;              { for ansi/wide strings, we need at least one register }              if is_ansistring(left.resultdef) or                is_widestring(left.resultdef) or              { ... as well as for dynamic arrays }                is_dynamic_array(left.resultdef) then                registersint:=max(registersint,1);           end         else           begin              { this rules are suboptimal, but they should give }              { good results                                }              registersint:=max(left.registersint,right.registersint);              { for ansi/wide strings, we need at least one register }              if is_ansistring(left.resultdef) or                is_widestring(left.resultdef) or              { ... as well as for dynamic arrays }                is_dynamic_array(left.resultdef) then                registersint:=max(registersint,1);              { need we an extra register when doing the restore ? }              if (left.registersint<=right.registersint) and              { only if the node needs less than 3 registers }              { two for the right node and one for the       }              { left address                             }                (registersint<3) then                inc(registersint);              { need we an extra register for the index ? }              if (right.expectloc<>LOC_REGISTER)              { only if the right node doesn't need a register }                and (right.registersint<1) then                inc(registersint);              { not correct, but what works better ?              if left.registersint>0 then                registersint:=max(registersint,2)              else                 min. one register                registersint:=max(registersint,1);              }           end;         registersfpu:=max(left.registersfpu,right.registersfpu);{$ifdef SUPPORT_MMX}         registersmmx:=max(left.registersmmx,right.registersmmx);{$endif SUPPORT_MMX}         if (not is_packed_array(left.resultdef)) or            ((tarraydef(left.resultdef).elepackedbitsize mod 8) = 0) then           if left.expectloc=LOC_CREFERENCE then             expectloc:=LOC_CREFERENCE           else             expectloc:=LOC_REFERENCE         else           if left.expectloc=LOC_CREFERENCE then             expectloc:=LOC_CSUBSETREF           else             expectloc:=LOC_SUBSETREF;      end;{*****************************************************************************                               TWITHNODE*****************************************************************************}    constructor twithnode.create(l:tnode);      begin         inherited create(withn,l);         fileinfo:=l.fileinfo;      end;    destructor twithnode.destroy;      begin        inherited destroy;      end;    constructor twithnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);      begin        inherited ppuload(t,ppufile);      end;    procedure twithnode.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);      end;    function twithnode.dogetcopy : tnode;      var         p : twithnode;      begin         p:=twithnode(inherited dogetcopy);         result:=p;      end;    function twithnode.pass_typecheck:tnode;      begin        result:=nil;        resultdef:=voidtype;        if assigned(left) then          typecheckpass(left);      end;    function twithnode.pass_1 : tnode;      begin        result:=nil;        expectloc:=LOC_VOID;        registersint:=left.registersint;        registersfpu:=left.registersfpu;{$ifdef SUPPORT_MMX}        registersmmx:=left.registersmmx;{$endif SUPPORT_MMX}      end;    function twithnode.docompare(p: tnode): boolean;      begin        docompare :=          inherited docompare(p);      end;    function is_big_untyped_addrnode(p: tnode): boolean;      begin        is_big_untyped_addrnode:=(p.nodetype=addrn) and 	  not (nf_typedaddr in p.flags) and (taddrnode(p).left.resultdef.size > 1);      end;begin  cloadvmtaddrnode := tloadvmtaddrnode;  caddrnode := taddrnode;  cderefnode := tderefnode;  csubscriptnode := tsubscriptnode;  cvecnode := tvecnode;  cwithnode := twithnode;end.
 |