| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229 | {    Copyright (c) 2000-2002 by Florian Klaempfl    Type checking and register allocation for load/assignment 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 nld;{$i fpcdefs.inc}interface    uses       node,       {$ifdef state_tracking}       nstate,       {$endif}       symconst,symbase,symtype,symsym,symdef;    type       Trttidatatype=(rdt_normal,rdt_o2s,rdt_s2o);       tloadnode = class(tunarynode)       protected          procdef : tprocdef;          procdefderef : tderef;       public          symtableentry : tsym;          symtableentryderef : tderef;          symtable : TSymtable;          constructor create(v : tsym;st : TSymtable);virtual;          constructor create_procvar(v : tsym;d:tprocdef;st : TSymtable);virtual;          constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderefimpl;override;          procedure derefimpl;override;          procedure set_mp(p:tnode);          function  is_addr_param_load:boolean;          function  dogetcopy : tnode;override;          function  pass_1 : tnode;override;          function  pass_typecheck:tnode;override;          procedure mark_write;override;          function  docompare(p: tnode): boolean; override;          procedure printnodedata(var t:text);override;          procedure setprocdef(p : tprocdef);       end;       tloadnodeclass = class of tloadnode;       { different assignment types }       tassigntype = (at_normal,at_plus,at_minus,at_star,at_slash);       tassignmentnode = class(tbinarynode)          assigntype : tassigntype;          constructor create(l,r : tnode);virtual;          constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function dogetcopy : tnode;override;          function pass_1 : tnode;override;          function pass_typecheck:tnode;override;       {$ifdef state_tracking}          function track_state_pass(exec_known:boolean):boolean;override;       {$endif state_tracking}          function docompare(p: tnode): boolean; override;       end;       tassignmentnodeclass = class of tassignmentnode;       tarrayconstructorrangenode = class(tbinarynode)          constructor create(l,r : tnode);virtual;          function pass_1 : tnode;override;          function pass_typecheck:tnode;override;       end;       tarrayconstructorrangenodeclass = class of tarrayconstructorrangenode;       tarrayconstructornode = class(tbinarynode)          constructor create(l,r : tnode);virtual;          function dogetcopy : tnode;override;          function pass_1 : tnode;override;          function pass_typecheck:tnode;override;          function docompare(p: tnode): boolean; override;          procedure force_type(def:tdef);          procedure insert_typeconvs;       end;       tarrayconstructornodeclass = class of tarrayconstructornode;       ttypenode = class(tnode)          allowed : boolean;          typedef : tdef;          typedefderef : tderef;          constructor create(def:tdef);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;          function docompare(p: tnode): boolean; override;       end;       ttypenodeclass = class of ttypenode;       trttinode = class(tnode)          l1,l2  : longint;          rttitype : trttitype;          rttidef : tstoreddef;          rttidefderef : tderef;          rttidatatype : Trttidatatype;          constructor create(def:tstoreddef;rt:trttitype;dt:Trttidatatype);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 pass_typecheck:tnode;override;          function docompare(p: tnode): boolean; override;       end;       trttinodeclass = class of trttinode;    var       cloadnode : tloadnodeclass;       cassignmentnode : tassignmentnodeclass;       carrayconstructorrangenode : tarrayconstructorrangenodeclass;       carrayconstructornode : tarrayconstructornodeclass;       ctypenode : ttypenodeclass;       crttinode : trttinodeclass;       { Current assignment node }       aktassignmentnode : tassignmentnode;implementation    uses      cutils,verbose,globtype,globals,systems,      symnot,      defutil,defcmp,      htypechk,pass_1,procinfo,paramgr,      ncon,ninl,ncnv,nmem,ncal,nutils,nbas,      cgobj,cgbase      ;{*****************************************************************************                             TLOADNODE*****************************************************************************}    constructor tloadnode.create(v : tsym;st : TSymtable);      begin         inherited create(loadn,nil);         if not assigned(v) then          internalerror(200108121);         symtableentry:=v;         symtable:=st;         procdef:=nil;      end;    constructor tloadnode.create_procvar(v : tsym;d:tprocdef;st : TSymtable);      begin         inherited create(loadn,nil);         if not assigned(v) then          internalerror(200108121);         symtableentry:=v;         symtable:=st;         procdef:=d;      end;    constructor tloadnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);      begin        inherited ppuload(t,ppufile);        ppufile.getderef(symtableentryderef);        symtable:=nil;        ppufile.getderef(procdefderef);      end;    procedure tloadnode.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);        ppufile.putderef(symtableentryderef);        ppufile.putderef(procdefderef);      end;    procedure tloadnode.buildderefimpl;      begin        inherited buildderefimpl;        symtableentryderef.build(symtableentry);        procdefderef.build(procdef);      end;    procedure tloadnode.derefimpl;      begin        inherited derefimpl;        symtableentry:=tsym(symtableentryderef.resolve);        symtable:=symtableentry.owner;        procdef:=tprocdef(procdefderef.resolve);      end;    procedure tloadnode.set_mp(p:tnode);      begin        { typen nodes should not be set }        if p.nodetype=typen then          internalerror(200301042);        left:=p;      end;    function tloadnode.dogetcopy : tnode;      var         n : tloadnode;      begin         n:=tloadnode(inherited dogetcopy);         n.symtable:=symtable;         n.symtableentry:=symtableentry;         n.procdef:=procdef;         result:=n;      end;    function tloadnode.is_addr_param_load:boolean;      begin        result:=(symtable.symtabletype=parasymtable) and                (symtableentry.typ=paravarsym) and                not(vo_has_local_copy in tparavarsym(symtableentry).varoptions) and                not(nf_load_self_pointer in flags) and                paramanager.push_addr_param(tparavarsym(symtableentry).varspez,tparavarsym(symtableentry).vardef,tprocdef(symtable.defowner).proccalloption);      end;    function tloadnode.pass_typecheck:tnode;      begin         result:=nil;         case symtableentry.typ of           absolutevarsym :             resultdef:=tabsolutevarsym(symtableentry).vardef;           constsym:             begin               if tconstsym(symtableentry).consttyp=constresourcestring then                 resultdef:=cansistringtype               else                 internalerror(22799);             end;           staticvarsym :             begin               tabstractvarsym(symtableentry).IncRefCountBy(1);               { static variables referenced in procedures or from finalization,                 variable needs to be in memory.                 It is too hard and the benefit is too small to detect whether a                 variable is only used in the finalization to add support for it (PFV) }               if assigned(current_procinfo) and                  (symtable.symtabletype=staticsymtable) and                  (                    (symtable.symtablelevel<>current_procinfo.procdef.localst.symtablelevel) or                    (current_procinfo.procdef.proctypeoption=potype_unitfinalize)                  ) then                 make_not_regable(self,vr_none);               resultdef:=tabstractvarsym(symtableentry).vardef;             end;           paravarsym,           localvarsym :             begin               tabstractvarsym(symtableentry).IncRefCountBy(1);               { Nested variable? The we need to load the framepointer of                 the parent procedure }               if assigned(current_procinfo) and                  (symtable.symtabletype in [localsymtable,parasymtable]) and                  (symtable.symtablelevel<>current_procinfo.procdef.parast.symtablelevel) then                 begin                   if assigned(left) then                     internalerror(200309289);                   left:=cloadparentfpnode.create(tprocdef(symtable.defowner));                   { we can't inline the referenced parent procedure }                   exclude(tprocdef(symtable.defowner).procoptions,po_inline);                   { reference in nested procedures, variable needs to be in memory }                   make_not_regable(self,vr_none);                 end;               { fix self type which is declared as voidpointer in the                 definition }               if vo_is_self in tabstractvarsym(symtableentry).varoptions then                 begin                   resultdef:=tprocdef(symtableentry.owner.defowner)._class;                   if (po_classmethod in tprocdef(symtableentry.owner.defowner).procoptions) or                      (po_staticmethod in tprocdef(symtableentry.owner.defowner).procoptions) then                     resultdef:=tclassrefdef.create(resultdef)                   else if is_object(resultdef) and                           (nf_load_self_pointer in flags) then                     resultdef:=tpointerdef.create(resultdef);                 end               else if vo_is_vmt in tabstractvarsym(symtableentry).varoptions then                 begin                   resultdef:=tprocdef(symtableentry.owner.defowner)._class;                   resultdef:=tclassrefdef.create(resultdef);                 end               else                 resultdef:=tabstractvarsym(symtableentry).vardef;             end;           procsym :             begin               { Return the first procdef. In case of overlaoded                 procdefs the matching procdef will be choosen                 when the expected procvardef is known, see get_information                 in htypechk.pas (PFV) }               if not assigned(procdef) then                 procdef:=tprocdef(tprocsym(symtableentry).ProcdefList[0])               else if po_kylixlocal in procdef.procoptions then                 CGMessage(type_e_cant_take_address_of_local_subroutine);               { the result is a procdef, addrn and proc_to_procvar                 typeconvn need this as resultdef so they know                 that the address needs to be returned }               resultdef:=procdef;               { process methodpointer }               if assigned(left) then                 typecheckpass(left);             end;           labelsym:             resultdef:=voidtype;           else             internalerror(200104141);         end;      end;    procedure Tloadnode.mark_write;    begin      include(flags,nf_write);    end;    function tloadnode.pass_1 : tnode;      begin         result:=nil;         expectloc:=LOC_REFERENCE;         registersint:=0;         registersfpu:=0;{$ifdef SUPPORT_MMX}         registersmmx:=0;{$endif SUPPORT_MMX}         if (cs_create_pic in current_settings.moduleswitches) and           not(symtableentry.typ in [paravarsym,localvarsym]) then           include(current_procinfo.flags,pi_needs_got);         case symtableentry.typ of            absolutevarsym :              ;            constsym:              begin                 if tconstsym(symtableentry).consttyp=constresourcestring then                   expectloc:=LOC_CREFERENCE;              end;            staticvarsym,            localvarsym,            paravarsym :              begin                if assigned(left) then                  firstpass(left);                if not is_addr_param_load and                   tabstractvarsym(symtableentry).is_regvar(is_addr_param_load) then                  expectloc:=tvarregable2tcgloc[tabstractvarsym(symtableentry).varregable]                else                  if (tabstractvarsym(symtableentry).varspez=vs_const) then                    expectloc:=LOC_CREFERENCE;                { we need a register for call by reference parameters }                if paramanager.push_addr_param(tabstractvarsym(symtableentry).varspez,tabstractvarsym(symtableentry).vardef,pocall_default) then                  registersint:=1;                if ([vo_is_thread_var,vo_is_dll_var]*tabstractvarsym(symtableentry).varoptions)<>[] then                  registersint:=1;                if (target_info.system=system_powerpc_darwin) and                   ([vo_is_dll_var,vo_is_external] * tabstractvarsym(symtableentry).varoptions <> []) then                  include(current_procinfo.flags,pi_needs_got);                { call to get address of threadvar }                if (vo_is_thread_var in tabstractvarsym(symtableentry).varoptions) then                  include(current_procinfo.flags,pi_do_call);                if nf_write in flags then                  Tabstractvarsym(symtableentry).trigger_notifications(vn_onwrite)                else                  Tabstractvarsym(symtableentry).trigger_notifications(vn_onread);                { count variable references }                if cg.t_times>1 then                  tabstractvarsym(symtableentry).IncRefCountBy(cg.t_times-1);              end;            procsym :                begin                   { method pointer ? }                   if assigned(left) then                     begin                        expectloc:=LOC_CREFERENCE;                        firstpass(left);                        registersint:=max(registersint,left.registersint);                        registersfpu:=max(registersfpu,left.registersfpu); {$ifdef SUPPORT_MMX}                        registersmmx:=max(registersmmx,left.registersmmx); {$endif SUPPORT_MMX}                     end;                end;           labelsym :             ;           else             internalerror(200104143);         end;      end;    function tloadnode.docompare(p: tnode): boolean;      begin        docompare :=          inherited docompare(p) and          (symtableentry = tloadnode(p).symtableentry) and          (procdef = tloadnode(p).procdef) and          (symtable = tloadnode(p).symtable);      end;    procedure tloadnode.printnodedata(var t:text);      begin        inherited printnodedata(t);        write(t,printnodeindention,'symbol = ',symtableentry.name);        if symtableentry.typ=procsym then          write(t,printnodeindention,'procdef = ',procdef.mangledname);        writeln(t,'');      end;    procedure tloadnode.setprocdef(p : tprocdef);      begin        procdef:=p;        resultdef:=p;        if po_local in p.procoptions then          CGMessage(type_e_cant_take_address_of_local_subroutine);      end;{*****************************************************************************                             TASSIGNMENTNODE*****************************************************************************}    constructor tassignmentnode.create(l,r : tnode);      begin         inherited create(assignn,l,r);         l.mark_write;         assigntype:=at_normal;      end;    constructor tassignmentnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);      begin        inherited ppuload(t,ppufile);        assigntype:=tassigntype(ppufile.getbyte);      end;    procedure tassignmentnode.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);        ppufile.putbyte(byte(assigntype));      end;    function tassignmentnode.dogetcopy : tnode;      var         n : tassignmentnode;      begin         n:=tassignmentnode(inherited dogetcopy);         n.assigntype:=assigntype;         result:=n;      end;    function tassignmentnode.pass_typecheck:tnode;      var        hp : tnode;        useshelper : boolean;      begin        result:=nil;        resultdef:=voidtype;        { must be made unique }        set_unique(left);        typecheckpass(left);{$ifdef old_append_str}        if is_ansistring(left.resultdef) then          begin            { fold <ansistring>:=<ansistring>+<char|shortstring|ansistring> }            if (right.nodetype=addn) and               left.isequal(tbinarynode(right).left) and               { don't fold multiple concatenations else we could get trouble                 with multiple uses of s               }               (tbinarynode(right).left.nodetype<>addn) and               (tbinarynode(right).right.nodetype<>addn) then              begin                { don't do a typecheckpass(right), since then the addnode }                { may insert typeconversions that make this optimization   }                { opportunity quite difficult to detect (JM)               }                typecheckpass(tbinarynode(right).left);                typecheckpass(tbinarynode(right).right);                if (tbinarynode(right).right.nodetype=stringconstn) or		   is_char(tbinarynode(right).right.resultdef) or                   is_shortstring(tbinarynode(right).right.resultdef) or                   is_ansistring(tbinarynode(right).right.resultdef) then                  begin                    { remove property flag so it'll not trigger an error }                    exclude(left.flags,nf_isproperty);                    { generate call to helper }                    hp:=ccallparanode.create(tbinarynode(right).right,                      ccallparanode.create(left,nil));                    if is_char(tbinarynode(right).right.resultdef) then                      result:=ccallnode.createintern('fpc_'+Tstringdef(left.resultdef).stringtypname+'_append_char',hp)                    else if is_shortstring(tbinarynode(right).right.resultdef) then                      result:=ccallnode.createintern('fpc_'+Tstringdef(left.resultdef).stringtypname+'_append_shortstring',hp)                    else                      result:=ccallnode.createintern('fpc_'+Tstringdef(left.resultdef).stringtypname+'_append_ansistring',hp);                    tbinarynode(right).right:=nil;                    left:=nil;                    exit;                 end;              end;          end        else         if is_shortstring(left.resultdef) then          begin            { fold <shortstring>:=<shortstring>+<shortstring>,              <shortstring>+<char> is handled by an optimized node }            if (right.nodetype=addn) and               left.isequal(tbinarynode(right).left) and               { don't fold multiple concatenations else we could get trouble                 with multiple uses of s }               (tbinarynode(right).left.nodetype<>addn) and               (tbinarynode(right).right.nodetype<>addn) then              begin                { don't do a typecheckpass(right), since then the addnode }                { may insert typeconversions that make this optimization   }                { opportunity quite difficult to detect (JM)               }                typecheckpass(tbinarynode(right).left);                typecheckpass(tbinarynode(right).right);                if is_shortstring(tbinarynode(right).right.resultdef) then                  begin                    { remove property flag so it'll not trigger an error }                    exclude(left.flags,nf_isproperty);                    { generate call to helper }                    hp:=ccallparanode.create(tbinarynode(right).right,                      ccallparanode.create(left,nil));                    if is_shortstring(tbinarynode(right).right.resultdef) then                      result:=ccallnode.createintern('fpc_shortstr_append_shortstr',hp);                    tbinarynode(right).right:=nil;                    left:=nil;                    exit;                 end;              end;          end;{$endif old_append_str}        typecheckpass(right);        set_varstate(right,vs_read,[vsf_must_be_valid]);        set_varstate(left,vs_written,[]);        if codegenerror then          exit;        { tp procvar support, when we don't expect a procvar          then we need to call the procvar }        if (left.resultdef.typ<>procvardef) then          maybe_call_procvar(right,true);        { assignments to formaldefs and open arrays aren't allowed }        if (left.resultdef.typ=formaldef) or           is_open_array(left.resultdef) then          CGMessage(type_e_assignment_not_allowed);        { test if node can be assigned, properties are allowed }        valid_for_assignment(left,true);        { assigning nil to a dynamic array clears the array }        if is_dynamic_array(left.resultdef) and           (right.nodetype=niln) then         begin           hp:=ccallparanode.create(caddrnode.create_internal                   (crttinode.create(tstoreddef(left.resultdef),initrtti,rdt_normal)),               ccallparanode.create(ctypeconvnode.create_internal(left,voidpointertype),nil));           result := ccallnode.createintern('fpc_dynarray_clear',hp);           left:=nil;           exit;         end;        { shortstring helpers can do the conversion directly,          so treat them separatly }        if (is_shortstring(left.resultdef)) then         begin           { insert typeconv, except for chars that are handled in             secondpass and except for ansi/wide string that can             be converted immediatly }           if not(is_char(right.resultdef) or                  (right.resultdef.typ=stringdef)) then             inserttypeconv(right,left.resultdef);           if right.resultdef.typ=stringdef then            begin              useshelper:=true;              { convert constant strings to shortstrings. But                skip empty constant strings, that will be handled                in secondpass }              if (right.nodetype=stringconstn) then                begin                  { verify if range fits within shortstring }                  { just emit a warning, delphi gives an    }                  { error, only if the type definition of   }                  { of the string is less  < 255 characters }                  if not is_open_string(left.resultdef) and                     (tstringconstnode(right).len > tstringdef(left.resultdef).len) then                     cgmessage(type_w_string_too_long);                  inserttypeconv(right,left.resultdef);                  if (tstringconstnode(right).len=0) then                    useshelper:=false;                end;             { rest is done in pass 1 (JM) }             if useshelper then               exit;            end         end        else          begin           { check if the assignment may cause a range check error }           check_ranges(fileinfo,right,left.resultdef);           inserttypeconv(right,left.resultdef);          end;        { call helpers for interface }        if is_interfacecom(left.resultdef) then         begin           if right.resultdef.is_related(left.resultdef) then             begin               hp:=                 ccallparanode.create(                   ctypeconvnode.create_internal(right,voidpointertype),                 ccallparanode.create(                   ctypeconvnode.create_internal(left,voidpointertype),                   nil));               result:=ccallnode.createintern('fpc_intf_assign',hp)             end           else             begin               hp:=                 ccallparanode.create(                   cguidconstnode.create(tobjectdef(left.resultdef).iidguid^),                 ccallparanode.create(                   ctypeconvnode.create_internal(right,voidpointertype),                 ccallparanode.create(                   ctypeconvnode.create_internal(left,voidpointertype),                   nil)));               result:=ccallnode.createintern('fpc_intf_assign_by_iid',hp);             end;           left:=nil;           right:=nil;           exit;         end;        { call helpers for variant, they can contain non ref. counted types like          vararrays which must be really copied }        if left.resultdef.typ=variantdef then         begin           hp:=ccallparanode.create(ctypeconvnode.create_internal(                 caddrnode.create_internal(right),voidpointertype),               ccallparanode.create(ctypeconvnode.create_internal(                 caddrnode.create_internal(left),voidpointertype),               nil));           result:=ccallnode.createintern('fpc_variant_copy',hp);           left:=nil;           right:=nil;           exit;         end;        { call helpers for windows widestrings, they aren't ref. counted }        if (tf_winlikewidestring in target_info.flags) and is_widestring(left.resultdef) then         begin           hp:=ccallparanode.create(ctypeconvnode.create_internal(right,voidpointertype),               ccallparanode.create(ctypeconvnode.create_internal(left,voidpointertype),               nil));           result:=ccallnode.createintern('fpc_widestr_assign',hp);           left:=nil;           right:=nil;           exit;         end;        { check if local proc/func is assigned to procvar }        if right.resultdef.typ=procvardef then          test_local_to_procvar(tprocvardef(right.resultdef),left.resultdef);      end;    function tassignmentnode.pass_1 : tnode;      var        hp: tnode;        oldassignmentnode : tassignmentnode;      begin         result:=nil;         expectloc:=LOC_VOID;         firstpass(left);         { Optimize the reuse of the destination of the assingment in left.           Allow the use of the left inside the tree generated on the right.           This is especially usefull for string routines where the destination           is pushed as a parameter. Using the final destination of left directly           save a temp allocation and copy of data (PFV) }         oldassignmentnode:=aktassignmentnode;         if right.nodetype=addn then           aktassignmentnode:=self         else           aktassignmentnode:=nil;         firstpass(right);         aktassignmentnode:=oldassignmentnode;         if nf_assign_done_in_right in flags then           begin             result:=right;             right:=nil;             exit;           end;         if codegenerror then           exit;         if (cs_opt_level1 in current_settings.optimizerswitches) and            (right.nodetype = calln) and            (right.resultdef=left.resultdef) and            { left must be a temp, since otherwise as soon as you modify the }            { result, the current left node is modified and that one may     }            { still be an argument to the function or even accessed in the   }            { function                                                       }            (             (              (left.nodetype = temprefn) and              paramanager.ret_in_param(right.resultdef,tcallnode(right).procdefinition.proccalloption)             ) or             { there's special support for ansi/widestrings in the callnode }             is_ansistring(right.resultdef) or             is_widestring(right.resultdef)            )  then           begin             make_not_regable(left,vr_addr);             tcallnode(right).funcretnode := left;             result := right;             left := nil;             right := nil;             exit;           end;         { assignment to refcounted variable -> inc/decref }         if (not is_class(left.resultdef) and            left.resultdef.needs_inittable) then           include(current_procinfo.flags,pi_do_call);        if (is_shortstring(left.resultdef)) then          begin           if right.resultdef.typ=stringdef then            begin              if (right.nodetype<>stringconstn) or                 (tstringconstnode(right).len<>0) then               begin{$ifdef old_append_str}                 if (cs_opt_level1 in current_settings.optimizerswitches) and                    (right.nodetype in [calln,blockn]) and                    (left.nodetype = temprefn) and                    is_shortstring(right.resultdef) and                    not is_open_string(left.resultdef) and                    (tstringdef(left.resultdef).len = 255) then                   begin                     { the blocknode case is handled in pass_generate_code at the temp }                     { reference level (mainly for callparatemp)  (JM)     }                     if (right.nodetype = calln) then                       begin                         tcallnode(right).funcretnode := left;                         result := right;                       end                     else                       exit;                   end                 else{$endif old_append_str}                   begin                     hp:=ccallparanode.create                           (right,                      ccallparanode.create(cinlinenode.create                           (in_high_x,false,left.getcopy),nil));                     result:=ccallnode.createinternreturn('fpc_'+tstringdef(right.resultdef).stringtypname+'_to_shortstr',hp,left);                     firstpass(result);                   end;                 left:=nil;                 right:=nil;                 exit;               end;            end;           end;         registersint:=left.registersint+right.registersint;         registersfpu:=max(left.registersfpu,right.registersfpu);{$ifdef SUPPORT_MMX}         registersmmx:=max(left.registersmmx,right.registersmmx);{$endif SUPPORT_MMX}      end;    function tassignmentnode.docompare(p: tnode): boolean;      begin        docompare :=          inherited docompare(p) and          (assigntype = tassignmentnode(p).assigntype);      end;{$ifdef state_tracking}    function Tassignmentnode.track_state_pass(exec_known:boolean):boolean;    var se:Tstate_entry;    begin        track_state_pass:=false;        if exec_known then            begin                track_state_pass:=right.track_state_pass(exec_known);                {Force a new resultdef pass.}                right.resultdef:=nil;                do_typecheckpass(right);                typecheckpass(right);                aktstate.store_fact(left.getcopy,right.getcopy);            end        else            aktstate.delete_fact(left);    end;{$endif}{*****************************************************************************                           TARRAYCONSTRUCTORRANGENODE*****************************************************************************}    constructor tarrayconstructorrangenode.create(l,r : tnode);      begin         inherited create(arrayconstructorrangen,l,r);      end;    function tarrayconstructorrangenode.pass_typecheck:tnode;      begin        result:=nil;        typecheckpass(left);        typecheckpass(right);        set_varstate(left,vs_read,[vsf_must_be_valid]);        set_varstate(right,vs_read,[vsf_must_be_valid]);        if codegenerror then         exit;        resultdef:=left.resultdef;      end;    function tarrayconstructorrangenode.pass_1 : tnode;      begin        firstpass(left);        firstpass(right);        expectloc:=LOC_CREFERENCE;        calcregisters(self,0,0,0);        result:=nil;      end;{****************************************************************************                            TARRAYCONSTRUCTORNODE*****************************************************************************}    constructor tarrayconstructornode.create(l,r : tnode);      begin         inherited create(arrayconstructorn,l,r);      end;    function tarrayconstructornode.dogetcopy : tnode;      var         n : tarrayconstructornode;      begin         n:=tarrayconstructornode(inherited dogetcopy);         result:=n;      end;    function tarrayconstructornode.pass_typecheck:tnode;      var        hdef  : tdef;        hp    : tarrayconstructornode;        len   : longint;        varia : boolean;      begin        result:=nil;      { are we allowing array constructor? Then convert it to a set.        Do this only if we didn't convert the arrayconstructor yet. This        is needed for the cases where the resultdef is forced for a second        run }        if (not allow_array_constructor) then         begin           hp:=tarrayconstructornode(getcopy);           arrayconstructor_to_set(tnode(hp));           result:=hp;           exit;         end;      { only pass left tree, right tree contains next construct if any }        hdef:=nil;        len:=0;        varia:=false;        if assigned(left) then         begin           hp:=self;           while assigned(hp) do            begin              typecheckpass(hp.left);              set_varstate(hp.left,vs_read,[vsf_must_be_valid]);              if (hdef=nil) then               hdef:=hp.left.resultdef              else               begin                 if (not varia) and (not equal_defs(hdef,hp.left.resultdef)) then                   begin                     { If both are integers we need to take the type that can hold both                       defs }                     if is_integer(hdef) and is_integer(hp.left.resultdef) then                       begin                         if is_in_limit(hdef,hp.left.resultdef) then                           hdef:=hp.left.resultdef;                       end                     else                       if (nf_novariaallowed in flags) then                         varia:=true;                   end;               end;              inc(len);              hp:=tarrayconstructornode(hp.right);            end;         end;         { Set the type of empty or varia arrays to void. Also           do this if the type is array of const/open array           because those can't be used with setelementdef }         if not assigned(hdef) or            varia or            is_array_of_const(hdef) or            is_open_array(hdef) then           hdef:=voidtype;         resultdef:=tarraydef.create(0,len-1,s32inttype);         tarraydef(resultdef).elementdef:=hdef;         include(tarraydef(resultdef).arrayoptions,ado_IsConstructor);         if varia then           include(tarraydef(resultdef).arrayoptions,ado_IsVariant);      end;    procedure tarrayconstructornode.force_type(def:tdef);      var        hp : tarrayconstructornode;      begin        tarraydef(resultdef).elementdef:=def;        include(tarraydef(resultdef).arrayoptions,ado_IsConstructor);        exclude(tarraydef(resultdef).arrayoptions,ado_IsVariant);        if assigned(left) then         begin           hp:=self;           while assigned(hp) do            begin              inserttypeconv(hp.left,def);              hp:=tarrayconstructornode(hp.right);            end;         end;      end;    procedure tarrayconstructornode.insert_typeconvs;      var        hp        : tarrayconstructornode;        dovariant : boolean;      begin        dovariant:=(nf_forcevaria in flags) or (ado_isvariant in tarraydef(resultdef).arrayoptions);        { only pass left tree, right tree contains next construct if any }        if assigned(left) then         begin           hp:=self;           while assigned(hp) do            begin              typecheckpass(hp.left);              { Insert typeconvs for array of const }              if dovariant then                { at this time C varargs are no longer an arrayconstructornode }                insert_varargstypeconv(hp.left,false);              hp:=tarrayconstructornode(hp.right);            end;         end;      end;    function tarrayconstructornode.pass_1 : tnode;      var        hp : tarrayconstructornode;        do_variant:boolean;      begin        do_variant:=(nf_forcevaria in flags) or (ado_isvariant in tarraydef(resultdef).arrayoptions);        result:=nil;        { Insert required type convs, this must be          done in pass 1, because the call must be          typecheckpassed already }        if assigned(left) then          begin            insert_typeconvs;            { call firstpass for all nodes }            hp:=self;            while assigned(hp) do              begin                if hp.left<>nil then                  begin                    {This check is pessimistic; a call will happen depending                     on the location in which the elements will be found in                     pass 2.}                    if not do_variant then                      include(current_procinfo.flags,pi_do_call);                    firstpass(hp.left);                  end;                hp:=tarrayconstructornode(hp.right);              end;          end;        expectloc:=LOC_CREFERENCE;        calcregisters(self,0,0,0);      end;    function tarrayconstructornode.docompare(p: tnode): boolean;    begin      docompare:=inherited docompare(p);    end;{*****************************************************************************                              TTYPENODE*****************************************************************************}    constructor ttypenode.create(def:tdef);      begin         inherited create(typen);         typedef:=def;         allowed:=false;      end;    constructor ttypenode.ppuload(t:tnodetype;ppufile:tcompilerppufile);      begin        inherited ppuload(t,ppufile);        ppufile.getderef(typedefderef);        allowed:=boolean(ppufile.getbyte);      end;    procedure ttypenode.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);        ppufile.putderef(typedefderef);        ppufile.putbyte(byte(allowed));      end;    procedure ttypenode.buildderefimpl;      begin        inherited buildderefimpl;        typedefderef.build(typedef);      end;    procedure ttypenode.derefimpl;      begin        inherited derefimpl;        typedef:=tdef(typedefderef.resolve);      end;    function ttypenode.pass_typecheck:tnode;      begin        result:=nil;        resultdef:=typedef;        { check if it's valid }        if typedef.typ = errordef then          CGMessage(parser_e_illegal_expression);      end;    function ttypenode.pass_1 : tnode;      begin         result:=nil;         expectloc:=LOC_VOID;         { a typenode can't generate code, so we give here           an error. Else it'll be an abstract error in pass_generate_code.           Only when the allowed flag is set we don't generate           an error }         if not allowed then          Message(parser_e_no_type_not_allowed_here);      end;    function ttypenode.dogetcopy : tnode;      var         n : ttypenode;      begin         n:=ttypenode(inherited dogetcopy);         n.allowed:=allowed;         n.typedef:=typedef;         result:=n;      end;    function ttypenode.docompare(p: tnode): boolean;      begin        docompare :=          inherited docompare(p);      end;{*****************************************************************************                              TRTTINODE*****************************************************************************}    constructor trttinode.create(def:tstoreddef;rt:trttitype;dt:Trttidatatype);      begin         inherited create(rttin);         rttidef:=def;         rttitype:=rt;         rttidatatype:=dt;      end;    constructor trttinode.ppuload(t:tnodetype;ppufile:tcompilerppufile);      begin        inherited ppuload(t,ppufile);        ppufile.getderef(rttidefderef);        rttitype:=trttitype(ppufile.getbyte);      end;    procedure trttinode.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);        ppufile.putderef(rttidefderef);        ppufile.putbyte(byte(rttitype));      end;    procedure trttinode.buildderefimpl;      begin        inherited buildderefimpl;        rttidefderef.build(rttidef);      end;    procedure trttinode.derefimpl;      begin        inherited derefimpl;        rttidef:=tstoreddef(rttidefderef.resolve);      end;    function trttinode.dogetcopy : tnode;      var         n : trttinode;      begin         n:=trttinode(inherited dogetcopy);         n.rttidef:=rttidef;         n.rttitype:=rttitype;         result:=n;      end;    function trttinode.pass_typecheck:tnode;      begin        { rtti information will be returned as a void pointer }        result:=nil;        resultdef:=voidpointertype;      end;    function trttinode.pass_1 : tnode;      begin        result:=nil;        expectloc:=LOC_CREFERENCE;      end;    function trttinode.docompare(p: tnode): boolean;      begin        docompare :=          inherited docompare(p) and          (rttidef = trttinode(p).rttidef) and          (rttitype = trttinode(p).rttitype);      end;begin   cloadnode:=tloadnode;   cassignmentnode:=tassignmentnode;   carrayconstructorrangenode:=tarrayconstructorrangenode;   carrayconstructornode:=tarrayconstructornode;   ctypenode:=ttypenode;   crttinode:=trttinode;end.
 |