| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910 | {    Copyright (c) 1998-2002 by Florian Klaempfl, Pierre Muller    Implementation for the symbols types of the symtable    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 symsym;{$i fpcdefs.inc}interface    uses       { common }       cutils,       { target }       globtype,globals,widestr,       { symtable }       symconst,symbase,symtype,symdef,defcmp,       { ppu }       ppu,finput,       cclasses,symnot,       { aasm }       aasmbase,       cpuinfo,cpubase,cgbase,cgutils,parabase       ;    type       { this class is the base for all symbol objects }       tstoredsym = class(tsym)       public          constructor create(st:tsymtyp;const n : string);          constructor ppuload(st:tsymtyp;ppufile:tcompilerppufile);          destructor destroy;override;          procedure ppuwrite(ppufile:tcompilerppufile);virtual;       end;       tlabelsym = class(tstoredsym)          used,          defined : boolean;          { points to the matching node, only valid resultdef pass is run and            the goto<->label relation in the node tree is created, should            be a tnode }          code : pointer;          { when the label is defined in an asm block, this points to the            generated asmlabel }          asmblocklabel : tasmlabel;          constructor create(const n : string);          constructor ppuload(ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          function mangledname:string;override;       end;       tunitsym = class(Tstoredsym)          module : tobject; { tmodule }          constructor create(const n : string;amodule : tobject);          constructor ppuload(ppufile:tcompilerppufile);          destructor destroy;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;       end;       terrorsym = class(Tsym)          constructor create;       end;       tprocsym = class(tstoredsym)       protected          FProcdefList   : TFPObjectList;          FProcdefDerefList : TFPList;       public          overloadchecked : boolean;          constructor create(const n : string);          constructor ppuload(ppufile:tcompilerppufile);          destructor destroy;override;          { writes all declarations except the specified one }          procedure write_parameter_lists(skipdef:tprocdef);          { tests, if all procedures definitions are defined and not }          { only forward                                             }          procedure check_forward;          procedure unchain_overload;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          procedure add_para_match_to(Aprocsym:Tprocsym;cpoptions:tcompare_paras_options);          function find_procdef_bytype(pt:Tproctypeoption):Tprocdef;          function find_procdef_bypara(para:TFPObjectList;retdef:tdef;cpoptions:tcompare_paras_options):Tprocdef;          function find_procdef_byprocvardef(d:Tprocvardef):Tprocdef;          function find_procdef_assignment_operator(fromdef,todef:tdef;var besteq:tequaltype):Tprocdef;          { currobjdef is the object def to assume, this is necessary for protected and            private, context is the object def we're really in, this is for the strict stuff }          function is_visible_for_object(currobjdef:tdef;context:tdef):boolean;override;          property ProcdefList:TFPObjectList read FProcdefList;       end;       ttypesym = class(Tstoredsym)          typedef      : tdef;          typedefderef : tderef;          constructor create(const n : string;def:tdef);          constructor ppuload(ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;       end;       tabstractvarsym = class(tstoredsym)          varoptions    : tvaroptions;          varspez       : tvarspez;  { sets the type of access }          varregable    : tvarregable;          varstate      : tvarstate;          notifications : Tlinkedlist;          constructor create(st:tsymtyp;const n : string;vsp:tvarspez;def:tdef;vopts:tvaroptions);          constructor ppuload(st:tsymtyp;ppufile:tcompilerppufile);          destructor  destroy;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          function  getsize : aint;          function  getpackedbitsize : longint;          function  is_regvar(refpara: boolean):boolean;          procedure trigger_notifications(what:Tnotification_flag);          function register_notification(flags:Tnotification_flags;                                         callback:Tnotification_callback):cardinal;          procedure unregister_notification(id:cardinal);        private          procedure setvardef(def:tdef);          _vardef     : tdef;          vardefderef : tderef;        public          property vardef: tdef read _vardef write setvardef;      end;      tfieldvarsym = class(tabstractvarsym)          fieldoffset   : aint;   { offset in record/object }          constructor create(const n : string;vsp:tvarspez;def:tdef;vopts:tvaroptions);          constructor ppuload(ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          function mangledname:string;override;      end;      tabstractnormalvarsym = class(tabstractvarsym)          defaultconstsym : tsym;          defaultconstsymderef : tderef;          localloc      : TLocation; { register/reference for local var }          initialloc    : TLocation; { initial location so it can still be initialized later after the location was changed by SSA }          constructor create(st:tsymtyp;const n : string;vsp:tvarspez;def:tdef;vopts:tvaroptions);          constructor ppuload(st:tsymtyp;ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;      end;      tlocalvarsym = class(tabstractnormalvarsym)          constructor create(const n : string;vsp:tvarspez;def:tdef;vopts:tvaroptions);          constructor ppuload(ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;      end;      tparavarsym = class(tabstractnormalvarsym)          paraloc       : array[tcallercallee] of TCGPara;          paranr        : word; { position of this parameter }{$ifdef EXTDEBUG}          eqval         : tequaltype;{$endif EXTDEBUG}          constructor create(const n : string;nr:word;vsp:tvarspez;def:tdef;vopts:tvaroptions);          constructor ppuload(ppufile:tcompilerppufile);          destructor destroy;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;      end;      tstaticvarsym = class(tabstractnormalvarsym)      private          _mangledname : pshortstring;      public          constructor create(const n : string;vsp:tvarspez;def:tdef;vopts:tvaroptions);          constructor create_dll(const n : string;vsp:tvarspez;def:tdef);          constructor create_C(const n,mangled : string;vsp:tvarspez;def:tdef);          constructor ppuload(ppufile:tcompilerppufile);          destructor destroy;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function mangledname:string;override;          procedure set_mangledname(const s:string);      end;      tabsolutevarsym = class(tabstractvarsym)      public         abstyp  : absolutetyp;{$ifdef i386}         absseg  : boolean;{$endif i386}         asmname : pshortstring;         addroffset : aint;         ref     : tpropaccesslist;         constructor create(const n : string;def:tdef);         constructor create_ref(const n : string;def:tdef;_ref:tpropaccesslist);         destructor  destroy;override;         constructor ppuload(ppufile:tcompilerppufile);         procedure buildderef;override;         procedure deref;override;         function  mangledname : string;override;         procedure ppuwrite(ppufile:tcompilerppufile);override;      end;       tpropaccesslisttypes=(palt_none,palt_read,palt_write,palt_stored);       tpropertysym = class(Tstoredsym)          propoptions   : tpropertyoptions;          overridenpropsym : tpropertysym;          overridenpropsymderef : tderef;          propdef       : tdef;          propdefderef  : tderef;          indexdef      : tdef;          indexdefderef : tderef;          index,          default        : longint;          propaccesslist : array[tpropaccesslisttypes] of tpropaccesslist;          constructor create(const n : string);          destructor  destroy;override;          constructor ppuload(ppufile:tcompilerppufile);          function  getsize : aint;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;       end;       tconstvalue = record         case integer of         0: (valueord : tconstexprint);         1: (valueordptr : tconstptruint);         2: (valueptr : pointer; len : longint);       end;       tconstsym = class(tstoredsym)          constdef    : tdef;          constdefderef : tderef;          consttyp    : tconsttyp;          value       : tconstvalue;          constructor create_ord(const n : string;t : tconsttyp;v : tconstexprint;def:tdef);          constructor create_ordptr(const n : string;t : tconsttyp;v : tconstptruint;def:tdef);          constructor create_ptr(const n : string;t : tconsttyp;v : pointer;def:tdef);          constructor create_string(const n : string;t : tconsttyp;str:pchar;l:longint);          constructor create_wstring(const n : string;t : tconsttyp;pw:pcompilerwidestring);          constructor ppuload(ppufile:tcompilerppufile);          destructor  destroy;override;          procedure buildderef;override;          procedure deref;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;       end;       tenumsym = class(Tstoredsym)          value      : longint;          definition : tenumdef;          definitionderef : tderef;          nextenum   : tenumsym;          constructor create(const n : string;def : tenumdef;v : longint);          constructor ppuload(ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          procedure order;       end;       tsyssym = class(Tstoredsym)          number : longint;          constructor create(const n : string;l : longint);          constructor ppuload(ppufile:tcompilerppufile);          destructor  destroy;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;       end;    const       maxmacrolen=16*1024;    type       pmacrobuffer = ^tmacrobuffer;       tmacrobuffer = array[0..maxmacrolen-1] of char;       tmacro = class(tstoredsym)          {Normally true, but false when a previously defined macro is undef-ed}          defined : boolean;          {True if this is a mac style compiler variable, in which case no macro           substitutions shall be done.}          is_compiler_var : boolean;          {Whether the macro was used. NOTE: A use of a macro which was never defined}          {e. g. an IFDEF which returns false, will not be registered as used,}          {since there is no place to register its use. }          is_used : boolean;          buftext : pchar;          buflen  : longint;          constructor create(const n : string);          constructor ppuload(ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          destructor  destroy;override;          function GetCopy:tmacro;       end;    var       generrorsym : tsym;implementation    uses       { global }       verbose,       { target }       systems,       { symtable }       defutil,symtable,       fmodule,       { tree }       node,       { aasm }       aasmtai,aasmdata,       { codegen }       paramgr,       procinfo       ;{****************************************************************************                               Helpers****************************************************************************}{****************************************************************************                          TSYM (base for all symtypes)****************************************************************************}    constructor tstoredsym.create(st:tsymtyp;const n : string);      begin         inherited create(st,n);         { Register in current_module }         if assigned(current_module) then           begin             current_module.symlist.Add(self);             SymId:=current_module.symlist.Count-1;           end;      end;    constructor tstoredsym.ppuload(st:tsymtyp;ppufile:tcompilerppufile);      begin         SymId:=ppufile.getlongint;         inherited Create(st,ppufile.getstring);         { Register symbol }         current_module.symlist[SymId]:=self;         ppufile.getposinfo(fileinfo);         ppufile.getsmallset(symoptions);      end;    procedure tstoredsym.ppuwrite(ppufile:tcompilerppufile);      begin         ppufile.putlongint(SymId);         ppufile.putstring(realname);         ppufile.putposinfo(fileinfo);         ppufile.putsmallset(symoptions);      end;    destructor tstoredsym.destroy;      begin        inherited destroy;      end;{****************************************************************************                                 TLABELSYM****************************************************************************}    constructor tlabelsym.create(const n : string);      begin         inherited create(labelsym,n);         used:=false;         defined:=false;         code:=nil;      end;    constructor tlabelsym.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(labelsym,ppufile);         code:=nil;         used:=false;         defined:=true;      end;    procedure tlabelsym.ppuwrite(ppufile:tcompilerppufile);      begin         if owner.symtabletype=globalsymtable then           Message(sym_e_ill_label_decl)         else           begin              inherited ppuwrite(ppufile);              ppufile.writeentry(iblabelsym);           end;      end;   function tlabelsym.mangledname:string;     begin       if not(defined) then         begin           defined:=true;           current_asmdata.getjumplabel(asmblocklabel);         end;       result:=asmblocklabel.name;     end;{****************************************************************************                                  TUNITSYM****************************************************************************}    constructor tunitsym.create(const n : string;amodule : tobject);      begin         inherited create(unitsym,n);         module:=amodule;      end;    constructor tunitsym.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(unitsym,ppufile);         module:=nil;      end;    destructor tunitsym.destroy;      begin         inherited destroy;      end;    procedure tunitsym.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.writeentry(ibunitsym);      end;{****************************************************************************                                  TPROCSYM****************************************************************************}    constructor tprocsym.create(const n : string);      begin         inherited create(procsym,n);         FProcdefList:=TFPObjectList.Create(false);         FProcdefderefList:=nil;         { the tprocdef have their own symoptions, make the procsym           always visible }         symoptions:=[sp_public];         overloadchecked:=false;      end;    constructor tprocsym.ppuload(ppufile:tcompilerppufile);      var         pdderef : tderef;         i,         pdcnt : longint;      begin         inherited ppuload(procsym,ppufile);         FProcdefList:=TFPObjectList.Create(false);         FProcdefDerefList:=TFPList.Create;         pdcnt:=ppufile.getword;         for i:=1 to pdcnt do          begin            ppufile.getderef(pdderef);            FProcdefDerefList.Add(Pointer(PtrInt(pdderef.dataidx)));          end;      end;    destructor tprocsym.destroy;      begin        FProcdefList.Free;        if assigned(FProcdefDerefList) then          FProcdefDerefList.Free;        inherited destroy;      end;    procedure tprocsym.ppuwrite(ppufile:tcompilerppufile);      var         i : longint;         d : tderef;      begin         inherited ppuwrite(ppufile);         ppufile.putword(FProcdefDerefList.Count);         for i:=0 to FProcdefDerefList.Count-1 do           begin             d.dataidx:=PtrInt(FProcdefDerefList[i]);             ppufile.putderef(d);           end;         ppufile.writeentry(ibprocsym);      end;    procedure tprocsym.write_parameter_lists(skipdef:tprocdef);      var        i  : longint;        pd : tprocdef;      begin        for i:=0 to ProcdefList.Count-1 do          begin            pd:=tprocdef(ProcdefList[i]);            if pd<>skipdef then              MessagePos1(pd.fileinfo,sym_h_param_list,pd.fullprocname(false));           end;      end;    procedure tprocsym.check_forward;      var        i  : longint;        pd : tprocdef;      begin        for i:=0 to ProcdefList.Count-1 do          begin            pd:=tprocdef(ProcdefList[i]);            if (pd.owner=owner) and (pd.forwarddef) then              begin                { For mode macpas. Make implicit externals (procedures declared in the interface                  section which do not have a counterpart in the implementation)                  to be an imported procedure }                if (m_mac in current_settings.modeswitches) and                   (pd.interfacedef) then                  begin                    pd.forwarddef:=false;                    pd.setmangledname(target_info.CPrefix+tprocdef(pd).procsym.realname);                  end                else                  begin                    MessagePos1(pd.fileinfo,sym_e_forward_not_resolved,pd.fullprocname(false));                    { Turn further error messages off }                    pd.forwarddef:=false;                  end                end;           end;      end;    procedure tprocsym.buildderef;      var        i  : longint;        pd : tprocdef;        d  : tderef;      begin        if not assigned(FProcdefDerefList) then          FProcdefDerefList:=TFPList.Create        else          FProcdefDerefList.Clear;        for i:=0 to ProcdefList.Count-1 do          begin            pd:=tprocdef(ProcdefList[i]);            { only write the proc definitions that belong              to this procsym and are in the global symtable }            if pd.owner=owner then              begin                d.build(pd);                FProcdefDerefList.Add(Pointer(PtrInt(d.dataidx)));              end;          end;      end;    procedure tprocsym.deref;      var        i  : longint;        pd : tprocdef;        d  : tderef;      begin        { Clear all procdefs }        ProcdefList.Clear;        if not assigned(FProcdefDerefList) then          internalerror(200611031);        for i:=0 to FProcdefDerefList.Count-1 do          begin            d.dataidx:=PtrInt(FProcdefDerefList[i]);            pd:=tprocdef(d.resolve);            ProcdefList.Add(pd);          end;      end;    procedure Tprocsym.add_para_match_to(Aprocsym:Tprocsym;cpoptions:tcompare_paras_options);      var        i  : longint;        pd : tprocdef;      begin        for i:=0 to ProcdefList.Count-1 do          begin            pd:=tprocdef(ProcdefList[i]);            if Aprocsym.find_procdef_bypara(pd.paras,nil,cpoptions)=nil then              Aprocsym.ProcdefList.Add(pd);          end;      end;    function Tprocsym.Find_procdef_bytype(pt:Tproctypeoption):Tprocdef;      var        i  : longint;        pd : tprocdef;      begin        result:=nil;        for i:=0 to ProcdefList.Count-1 do          begin            pd:=tprocdef(ProcdefList[i]);            if pd.proctypeoption=pt then              begin                result:=pd;                exit;              end;          end;      end;    function Tprocsym.Find_procdef_bypara(para:TFPObjectList;retdef:tdef;                                            cpoptions:tcompare_paras_options):Tprocdef;      var        i  : longint;        pd : tprocdef;        eq : tequaltype;      begin        result:=nil;        for i:=0 to ProcdefList.Count-1 do          begin            pd:=tprocdef(ProcdefList[i]);            if assigned(retdef) then              eq:=compare_defs(retdef,pd.returndef,nothingn)            else              eq:=te_equal;            if (eq>=te_equal) or               ((cpo_allowconvert in cpoptions) and (eq>te_incompatible)) then              begin                eq:=compare_paras(para,pd.paras,cp_value_equal_const,cpoptions);                if (eq>=te_equal) or                   ((cpo_allowconvert in cpoptions) and (eq>te_incompatible)) then                  begin                    result:=pd;                    exit;                  end;              end;          end;      end;    function Tprocsym.Find_procdef_byprocvardef(d:Tprocvardef):Tprocdef;      var        i  : longint;        bestpd,        pd : tprocdef;        eq,besteq : tequaltype;      begin        { This function will return the pprocdef of pprocsym that          is the best match for procvardef. When there are multiple          matches it returns nil.}        result:=nil;        bestpd:=nil;        besteq:=te_incompatible;        for i:=0 to ProcdefList.Count-1 do          begin            pd:=tprocdef(ProcdefList[i]);            eq:=proc_to_procvar_equal(pd,d);            if eq>=te_equal then              begin                { multiple procvars with the same equal level }                if assigned(bestpd) and                   (besteq=eq) then                  exit;                if eq>besteq then                  begin                    besteq:=eq;                    bestpd:=pd;                  end;              end;          end;        result:=bestpd;      end;    function Tprocsym.Find_procdef_assignment_operator(fromdef,todef:tdef;var besteq:tequaltype):Tprocdef;      var        paraidx,        i  : longint;        bestpd,        hpd,        pd : tprocdef;        convtyp : tconverttype;        eq      : tequaltype;      begin        { This function will return the pprocdef of pprocsym that          is the best match for procvardef. When there are multiple          matches it returns nil.}        result:=nil;        bestpd:=nil;        besteq:=te_incompatible;        for i:=0 to ProcdefList.Count-1 do          begin            pd:=tprocdef(ProcdefList[i]);            if equal_defs(todef,pd.returndef) and               { the result type must be always really equal and not an alias,                 if you mess with this code, check tw4093 }               ((todef=pd.returndef) or                (                  not(df_unique in todef.defoptions) and                  not(df_unique in pd.returndef.defoptions)                )               ) then              begin                paraidx:=0;                { ignore vs_hidden parameters }                while (paraidx<pd.paras.count) and                      assigned(pd.paras[paraidx]) and                      (vo_is_hidden_para in tparavarsym(pd.paras[paraidx]).varoptions) do                  inc(paraidx);                if (paraidx<pd.paras.count) and                   assigned(pd.paras[paraidx]) then                  begin                    eq:=compare_defs_ext(fromdef,tparavarsym(pd.paras[paraidx]).vardef,nothingn,convtyp,hpd,[]);                    { alias? if yes, only l1 choice,                      if you mess with this code, check tw4093 }                    if (eq=te_exact) and                       (fromdef<>tparavarsym(pd.paras[paraidx]).vardef) and                       ((df_unique in fromdef.defoptions) or                       (df_unique in tparavarsym(pd.paras[paraidx]).vardef.defoptions)) then                      eq:=te_convert_l1;                    if eq=te_exact then                      begin                        besteq:=eq;                        result:=pd;                        exit;                      end;                    if eq>besteq then                      begin                        bestpd:=pd;                        besteq:=eq;                      end;                  end;              end;          end;        result:=bestpd;      end;    procedure tprocsym.unchain_overload;      var        i  : longint;        pd : tprocdef;      begin        { remove all overloaded procdefs from the          procdeflist that are not in the current symtable }        overloadchecked:=false;        { reset new lists }        for i:=0 to ProcdefList.Count-1 do          begin            pd:=tprocdef(ProcdefList[i]);            { only keep the proc definitions:              - are not deref'd (def=nil)              - are in the same symtable as the procsym (for example both                are in the staticsymtable) }            if not(pd.owner=owner) then              ProcdefList[i]:=nil;          end;        { Remove cleared entries }        ProcdefList.Pack;      end;    function tprocsym.is_visible_for_object(currobjdef:tdef;context:tdef):boolean;      var        i  : longint;        pd : tprocdef;      begin        { This procsym is visible, when there is at least          one of the procdefs visible }        result:=false;        for i:=0 to ProcdefList.Count-1 do          begin            pd:=tprocdef(ProcdefList[i]);            if (pd.owner=owner) and                pd.is_visible_for_object(tobjectdef(currobjdef),tobjectdef(context)) then              begin                result:=true;                exit;              end;          end;      end;{****************************************************************************                                  TERRORSYM****************************************************************************}    constructor terrorsym.create;      begin        inherited create(errorsym,'');      end;{****************************************************************************                                TPROPERTYSYM****************************************************************************}    constructor tpropertysym.create(const n : string);      var        pap : tpropaccesslisttypes;      begin         inherited create(propertysym,n);         propoptions:=[];         index:=0;         default:=0;         propdef:=nil;         indexdef:=nil;         for pap:=low(tpropaccesslisttypes) to high(tpropaccesslisttypes) do           propaccesslist[pap]:=tpropaccesslist.create;      end;    constructor tpropertysym.ppuload(ppufile:tcompilerppufile);      var        pap : tpropaccesslisttypes;      begin         inherited ppuload(propertysym,ppufile);         ppufile.getsmallset(propoptions);         ppufile.getderef(overridenpropsymderef);         ppufile.getderef(propdefderef);         index:=ppufile.getlongint;         default:=ppufile.getlongint;         ppufile.getderef(indexdefderef);         for pap:=low(tpropaccesslisttypes) to high(tpropaccesslisttypes) do           propaccesslist[pap]:=ppufile.getpropaccesslist;      end;    destructor tpropertysym.destroy;      var        pap : tpropaccesslisttypes;      begin         for pap:=low(tpropaccesslisttypes) to high(tpropaccesslisttypes) do           propaccesslist[pap].free;         inherited destroy;      end;    procedure tpropertysym.buildderef;      var        pap : tpropaccesslisttypes;      begin        overridenpropsymderef.build(overridenpropsym);        propdefderef.build(propdef);        indexdefderef.build(indexdef);        for pap:=low(tpropaccesslisttypes) to high(tpropaccesslisttypes) do          propaccesslist[pap].buildderef;      end;    procedure tpropertysym.deref;      var        pap : tpropaccesslisttypes;      begin        overridenpropsym:=tpropertysym(overridenpropsymderef.resolve);        indexdef:=tdef(indexdefderef.resolve);        propdef:=tdef(propdefderef.resolve);        for pap:=low(tpropaccesslisttypes) to high(tpropaccesslisttypes) do          propaccesslist[pap].resolve;      end;    function tpropertysym.getsize : aint;      begin         getsize:=0;      end;    procedure tpropertysym.ppuwrite(ppufile:tcompilerppufile);      var        pap : tpropaccesslisttypes;      begin        inherited ppuwrite(ppufile);        ppufile.putsmallset(propoptions);        ppufile.putderef(overridenpropsymderef);        ppufile.putderef(propdefderef);        ppufile.putlongint(index);        ppufile.putlongint(default);        ppufile.putderef(indexdefderef);        for pap:=low(tpropaccesslisttypes) to high(tpropaccesslisttypes) do          ppufile.putpropaccesslist(propaccesslist[pap]);        ppufile.writeentry(ibpropertysym);      end;{****************************************************************************                            TABSTRACTVARSYM****************************************************************************}    constructor tabstractvarsym.create(st:tsymtyp;const n : string;vsp:tvarspez;def:tdef;vopts:tvaroptions);      begin         inherited create(st,n);         vardef:=def;         varspez:=vsp;         varstate:=vs_declared;         varoptions:=vopts;      end;    constructor tabstractvarsym.ppuload(st:tsymtyp;ppufile:tcompilerppufile);      begin         inherited ppuload(st,ppufile);         varstate:=vs_readwritten;         varspez:=tvarspez(ppufile.getbyte);         varregable:=tvarregable(ppufile.getbyte);         ppufile.getderef(vardefderef);         ppufile.getsmallset(varoptions);      end;    destructor tabstractvarsym.destroy;      begin        if assigned(notifications) then          notifications.destroy;        inherited destroy;      end;    procedure tabstractvarsym.buildderef;      begin        vardefderef.build(vardef);      end;    procedure tabstractvarsym.deref;      var        oldvarregable: tvarregable;      begin        { setting the vardef also updates varregable. We just loaded this }        { value from a ppu, so it must not be changed (e.g. tw7817a.pp/   }        { tw7817b.pp: the address is taken of a local variable in an      }        { inlined procedure -> must remain non-regable when inlining)     }        oldvarregable:=varregable;        vardef:=tdef(vardefderef.resolve);        varregable:=oldvarregable;      end;    procedure tabstractvarsym.ppuwrite(ppufile:tcompilerppufile);      var        oldintfcrc : boolean;      begin         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(varspez));         oldintfcrc:=ppufile.do_crc;         ppufile.do_crc:=false;         ppufile.putbyte(byte(varregable));         ppufile.do_crc:=oldintfcrc;         ppufile.putderef(vardefderef);         ppufile.putsmallset(varoptions);      end;    function tabstractvarsym.getsize : aint;      begin        if assigned(vardef) and           ((vardef.typ<>arraydef) or            is_dynamic_array(vardef) or            (tarraydef(vardef).highrange>=tarraydef(vardef).lowrange)) then          result:=vardef.size        else          result:=0;      end;    function  tabstractvarsym.getpackedbitsize : longint;      begin        { bitpacking is only done for ordinals }        if not is_ordinal(vardef) then          internalerror(2006082010);        result:=vardef.packedbitsize;      end;    function tabstractvarsym.is_regvar(refpara: boolean):boolean;      begin        { Register variables are not allowed in the following cases:           - regvars are disabled           - exceptions are used (after an exception is raised the contents of the               registers is not valid anymore)           - it has a local copy           - the value needs to be in memory (i.e. reference counted) }        result:=(cs_opt_regvar in current_settings.optimizerswitches) and                not(pi_has_assembler_block in current_procinfo.flags) and                not(pi_uses_exceptions in current_procinfo.flags) and                not(vo_has_local_copy in varoptions) and                ((refpara and                  (varregable <> vr_none)) or                 (not refpara and                  not(varregable in [vr_none,vr_addr]))){$if not defined(powerpc) and not defined(powerpc64)}                and ((vardef.typ <> recorddef) or                     (varregable = vr_addr) or                     not(varstate in [vs_written,vs_readwritten]));{$endif}      end;    procedure tabstractvarsym.trigger_notifications(what:Tnotification_flag);    var n:Tnotification;    begin        if assigned(notifications) then          begin            n:=Tnotification(notifications.first);            while assigned(n) do              begin                if what in n.flags then                  n.callback(what,self);                n:=Tnotification(n.next);              end;          end;    end;    function Tabstractvarsym.register_notification(flags:Tnotification_flags;callback:                                           Tnotification_callback):cardinal;    var n:Tnotification;    begin      if not assigned(notifications) then        notifications:=Tlinkedlist.create;      n:=Tnotification.create(flags,callback);      register_notification:=n.id;      notifications.concat(n);    end;    procedure Tabstractvarsym.unregister_notification(id:cardinal);    var n:Tnotification;    begin      if not assigned(notifications) then        internalerror(200212311)      else        begin            n:=Tnotification(notifications.first);            while assigned(n) do              begin                if n.id=id then                  begin                    notifications.remove(n);                    n.destroy;                    exit;                  end;                n:=Tnotification(n.next);              end;            internalerror(200212311)        end;    end;    procedure tabstractvarsym.setvardef(def:tdef);      begin        _vardef := def;         { can we load the value into a register ? }        if not assigned(owner) or           (owner.symtabletype in [localsymtable,parasymtable]) or           (            (owner.symtabletype=staticsymtable) and            not(cs_create_pic in current_settings.moduleswitches)           ) then          begin            if tstoreddef(vardef).is_intregable then              varregable:=vr_intreg            else{ $warning TODO: no fpu regvar in staticsymtable yet, need initialization with 0 }              if {(                  not assigned(owner) or                  (owner.symtabletype<>staticsymtable)                 ) and }                 tstoreddef(vardef).is_fpuregable then                 begin{$ifdef x86}                   if use_sse(vardef) then                     varregable:=vr_mmreg                   else{$else x86}                     varregable:=vr_fpureg;{$endif x86}                 end;          end;      end;{****************************************************************************                               TFIELDVARSYM****************************************************************************}    constructor tfieldvarsym.create(const n : string;vsp:tvarspez;def:tdef;vopts:tvaroptions);      begin         inherited create(fieldvarsym,n,vsp,def,vopts);         fieldoffset:=-1;      end;    constructor tfieldvarsym.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(fieldvarsym,ppufile);         fieldoffset:=ppufile.getaint;      end;    procedure tfieldvarsym.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putaint(fieldoffset);         ppufile.writeentry(ibfieldvarsym);      end;    function tfieldvarsym.mangledname:string;      var        srsym : tsym;        srsymtable : tsymtable;      begin        if sp_static in symoptions then          begin            if searchsym(lower(owner.name^)+'_'+name,srsym,srsymtable) then              result:=srsym.mangledname            { when generating the debug info for the module in which the }            { symbol is defined, the localsymtable of that module is     }            { already popped from the symtablestack                      }            else if searchsym_in_module(current_module,lower(owner.name^)+'_'+name,srsym,srsymtable) then              result:=srsym.mangledname            else              internalerror(2007012501);          end        else          result:=inherited mangledname;      end;{****************************************************************************                        TABSTRACTNORMALVARSYM****************************************************************************}    constructor tabstractnormalvarsym.create(st:tsymtyp;const n : string;vsp:tvarspez;def:tdef;vopts:tvaroptions);      begin         inherited create(st,n,vsp,def,vopts);         fillchar(localloc,sizeof(localloc),0);         fillchar(initialloc,sizeof(initialloc),0);         defaultconstsym:=nil;      end;    constructor tabstractnormalvarsym.ppuload(st:tsymtyp;ppufile:tcompilerppufile);      begin         inherited ppuload(st,ppufile);         fillchar(localloc,sizeof(localloc),0);         fillchar(initialloc,sizeof(initialloc),0);         ppufile.getderef(defaultconstsymderef);      end;    procedure tabstractnormalvarsym.buildderef;      begin        inherited buildderef;        defaultconstsymderef.build(defaultconstsym);      end;    procedure tabstractnormalvarsym.deref;      begin        inherited deref;        defaultconstsym:=tsym(defaultconstsymderef.resolve);      end;    procedure tabstractnormalvarsym.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putderef(defaultconstsymderef);      end;{****************************************************************************                             Tstaticvarsym****************************************************************************}    constructor tstaticvarsym.create(const n : string;vsp:tvarspez;def:tdef;vopts:tvaroptions);      begin         inherited create(staticvarsym,n,vsp,def,vopts);         _mangledname:=nil;      end;    constructor tstaticvarsym.create_dll(const n : string;vsp:tvarspez;def:tdef);      begin         tstaticvarsym(self).create(n,vsp,def,[vo_is_dll_var]);      end;    constructor tstaticvarsym.create_C(const n,mangled : string;vsp:tvarspez;def:tdef);      begin         tstaticvarsym(self).create(n,vsp,def,[]);         set_mangledname(mangled);      end;    constructor tstaticvarsym.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(staticvarsym,ppufile);         if vo_has_mangledname in varoptions then           _mangledname:=stringdup(ppufile.getstring)         else           _mangledname:=nil;      end;    destructor tstaticvarsym.destroy;      begin        if assigned(_mangledname) then          begin{$ifdef MEMDEBUG}            memmanglednames.start;{$endif MEMDEBUG}            stringdispose(_mangledname);{$ifdef MEMDEBUG}            memmanglednames.stop;{$endif MEMDEBUG}          end;        inherited destroy;      end;    procedure tstaticvarsym.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         if vo_has_mangledname in varoptions then           ppufile.putstring(_mangledname^);         ppufile.writeentry(ibstaticvarsym);      end;    function tstaticvarsym.mangledname:string;      var        prefix : string[2];      begin        if not assigned(_mangledname) then          begin            if (vo_is_typed_const in varoptions) then              prefix:='TC'            else              prefix:='U';      {$ifdef compress}            _mangledname:=stringdup(minilzw_encode(make_mangledname(prefix,owner,name)));      {$else}           _mangledname:=stringdup(make_mangledname(prefix,owner,name));      {$endif}          end;        result:=_mangledname^;      end;    procedure tstaticvarsym.set_mangledname(const s:string);      begin        stringdispose(_mangledname);      {$ifdef compress}        _mangledname:=stringdup(minilzw_encode(s));      {$else}        _mangledname:=stringdup(s);      {$endif}        include(varoptions,vo_has_mangledname);      end;{****************************************************************************                               TLOCALVARSYM****************************************************************************}    constructor tlocalvarsym.create(const n : string;vsp:tvarspez;def:tdef;vopts:tvaroptions);      begin         inherited create(localvarsym,n,vsp,def,vopts);      end;    constructor tlocalvarsym.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(localvarsym,ppufile);      end;    procedure tlocalvarsym.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.writeentry(iblocalvarsym);      end;{****************************************************************************                              TPARAVARSYM****************************************************************************}    constructor tparavarsym.create(const n : string;nr:word;vsp:tvarspez;def:tdef;vopts:tvaroptions);      begin         inherited create(paravarsym,n,vsp,def,vopts);         if (vsp in [vs_var,vs_value,vs_const]) then           varstate := vs_initialised;         paranr:=nr;         paraloc[calleeside].init;         paraloc[callerside].init;      end;    destructor tparavarsym.destroy;      begin        paraloc[calleeside].done;        paraloc[callerside].done;        inherited destroy;      end;    constructor tparavarsym.ppuload(ppufile:tcompilerppufile);      var        b : byte;      begin         inherited ppuload(paravarsym,ppufile);         paranr:=ppufile.getword;         { The var state of parameter symbols is fixed after writing them so           we write them to the unit file.           This enables constant folding for inline procedures loaded from units         }         varstate:=tvarstate(ppufile.getbyte);         paraloc[calleeside].init;         paraloc[callerside].init;         if vo_has_explicit_paraloc in varoptions then           begin             b:=ppufile.getbyte;             if b<>sizeof(paraloc[callerside].location^) then               internalerror(200411154);             ppufile.getdata(paraloc[callerside].add_location^,sizeof(paraloc[callerside].location^));             paraloc[callerside].size:=paraloc[callerside].location^.size;             paraloc[callerside].intsize:=tcgsize2size[paraloc[callerside].size];           end;      end;    procedure tparavarsym.ppuwrite(ppufile:tcompilerppufile);      var        oldintfcrc : boolean;      begin         inherited ppuwrite(ppufile);         ppufile.putword(paranr);         { The var state of parameter symbols is fixed after writing them so           we write them to the unit file.           This enables constant folding for inline procedures loaded from units         }         oldintfcrc:=ppufile.do_crc;         ppufile.do_crc:=false;         ppufile.putbyte(ord(varstate));         ppufile.do_crc:=oldintfcrc;         if vo_has_explicit_paraloc in varoptions then           begin             paraloc[callerside].check_simple_location;             ppufile.putbyte(sizeof(paraloc[callerside].location^));             ppufile.putdata(paraloc[callerside].location^,sizeof(paraloc[callerside].location^));           end;         ppufile.writeentry(ibparavarsym);      end;{****************************************************************************                               TABSOLUTEVARSYM****************************************************************************}    constructor tabsolutevarsym.create(const n : string;def:tdef);      begin        inherited create(absolutevarsym,n,vs_value,def,[]);        ref:=nil;      end;    constructor tabsolutevarsym.create_ref(const n : string;def:tdef;_ref:tpropaccesslist);      begin        inherited create(absolutevarsym,n,vs_value,def,[]);        ref:=_ref;      end;    destructor tabsolutevarsym.destroy;      begin        if assigned(ref) then          ref.free;        inherited destroy;      end;    constructor tabsolutevarsym.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(absolutevarsym,ppufile);         ref:=nil;         asmname:=nil;         abstyp:=absolutetyp(ppufile.getbyte);{$ifdef i386}         absseg:=false;{$endif i386}         case abstyp of           tovar :             ref:=ppufile.getpropaccesslist;           toasm :             asmname:=stringdup(ppufile.getstring);           toaddr :             begin               addroffset:=ppufile.getaint;{$ifdef i386}               absseg:=boolean(ppufile.getbyte);{$endif i386}             end;         end;      end;    procedure tabsolutevarsym.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(abstyp));         case abstyp of           tovar :             ppufile.putpropaccesslist(ref);           toasm :             ppufile.putstring(asmname^);           toaddr :             begin               ppufile.putaint(addroffset);{$ifdef i386}               ppufile.putbyte(byte(absseg));{$endif i386}             end;         end;         ppufile.writeentry(ibabsolutevarsym);      end;    procedure tabsolutevarsym.buildderef;      begin        inherited buildderef;        if (abstyp=tovar) then          ref.buildderef;      end;    procedure tabsolutevarsym.deref;      begin         inherited deref;         { own absolute deref }         if (abstyp=tovar) then           ref.resolve;      end;    function tabsolutevarsym.mangledname : string;      begin         case abstyp of           toasm :             mangledname:=asmname^;           toaddr :             mangledname:='$'+tostr(addroffset);           else             internalerror(200411062);         end;      end;{****************************************************************************                                  TCONSTSYM****************************************************************************}    constructor tconstsym.create_ord(const n : string;t : tconsttyp;v : tconstexprint;def:tdef);      begin         inherited create(constsym,n);         fillchar(value, sizeof(value), #0);         consttyp:=t;         value.valueord:=v;         constdef:=def;      end;    constructor tconstsym.create_ordptr(const n : string;t : tconsttyp;v : tconstptruint;def:tdef);      begin         inherited create(constsym,n);         fillchar(value, sizeof(value), #0);         consttyp:=t;         value.valueordptr:=v;         constdef:=def;      end;    constructor tconstsym.create_ptr(const n : string;t : tconsttyp;v : pointer;def:tdef);      begin         inherited create(constsym,n);         fillchar(value, sizeof(value), #0);         consttyp:=t;         value.valueptr:=v;         constdef:=def;      end;    constructor tconstsym.create_string(const n : string;t : tconsttyp;str:pchar;l:longint);      begin         inherited create(constsym,n);         fillchar(value, sizeof(value), #0);         consttyp:=t;         value.valueptr:=str;         constdef:=nil;         value.len:=l;      end;    constructor tconstsym.create_wstring(const n : string;t : tconsttyp;pw:pcompilerwidestring);      begin         inherited create(constsym,n);         fillchar(value, sizeof(value), #0);         consttyp:=t;         pcompilerwidestring(value.valueptr):=pw;         constdef:=nil;         value.len:=getlengthwidestring(pw);      end;    constructor tconstsym.ppuload(ppufile:tcompilerppufile);      var         pd : pbestreal;         ps : pnormalset;         pc : pchar;         pw : pcompilerwidestring;      begin         inherited ppuload(constsym,ppufile);         constdef:=nil;         consttyp:=tconsttyp(ppufile.getbyte);         fillchar(value, sizeof(value), #0);         case consttyp of           constord :             begin               ppufile.getderef(constdefderef);               value.valueord:=ppufile.getexprint;             end;           constpointer :             begin               ppufile.getderef(constdefderef);               value.valueordptr:=ppufile.getptruint;             end;           constwstring :             begin               initwidestring(pw);               setlengthwidestring(pw,ppufile.getlongint);               ppufile.getdata(pw^.data,pw^.len*sizeof(tcompilerwidechar));               pcompilerwidestring(value.valueptr):=pw;             end;           conststring,           constresourcestring :             begin               value.len:=ppufile.getlongint;               getmem(pc,value.len+1);               ppufile.getdata(pc^,value.len);               value.valueptr:=pc;             end;           constreal :             begin               new(pd);               pd^:=ppufile.getreal;               value.valueptr:=pd;             end;           constset :             begin               ppufile.getderef(constdefderef);               new(ps);               ppufile.getnormalset(ps^);               value.valueptr:=ps;             end;           constguid :             begin               new(pguid(value.valueptr));               ppufile.getdata(value.valueptr^,sizeof(tguid));             end;           constnil : ;           else             Message1(unit_f_ppu_invalid_entry,tostr(ord(consttyp)));         end;      end;    destructor tconstsym.destroy;      begin        case consttyp of          conststring,          constresourcestring :            freemem(pchar(value.valueptr),value.len+1);          constwstring :            donewidestring(pcompilerwidestring(value.valueptr));          constreal :            dispose(pbestreal(value.valueptr));          constset :            dispose(pnormalset(value.valueptr));          constguid :            dispose(pguid(value.valueptr));        end;        inherited destroy;      end;    procedure tconstsym.buildderef;      begin        if consttyp in [constord,constpointer,constset] then          constdefderef.build(constdef);      end;    procedure tconstsym.deref;      begin        if consttyp in [constord,constpointer,constset] then          constdef:=tdef(constdefderef.resolve);      end;    procedure tconstsym.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(consttyp));         case consttyp of           constnil : ;           constord :             begin               ppufile.putderef(constdefderef);               ppufile.putexprint(value.valueord);             end;           constpointer :             begin               ppufile.putderef(constdefderef);               ppufile.putptruint(value.valueordptr);             end;           constwstring :             begin               ppufile.putlongint(getlengthwidestring(pcompilerwidestring(value.valueptr)));               ppufile.putdata(pcompilerwidestring(value.valueptr)^.data,pcompilerwidestring(value.valueptr)^.len*sizeof(tcompilerwidechar));             end;           conststring,           constresourcestring :             begin               ppufile.putlongint(value.len);               ppufile.putdata(pchar(value.valueptr)^,value.len);             end;           constreal :             ppufile.putreal(pbestreal(value.valueptr)^);           constset :             begin               ppufile.putderef(constdefderef);               ppufile.putnormalset(value.valueptr^);             end;           constguid :             ppufile.putdata(value.valueptr^,sizeof(tguid));         else           internalerror(13);         end;        ppufile.writeentry(ibconstsym);      end;{****************************************************************************                                  TENUMSYM****************************************************************************}    constructor tenumsym.create(const n : string;def : tenumdef;v : longint);      begin         inherited create(enumsym,n);         definition:=def;         value:=v;         { First entry? Then we need to set the minval }         if def.firstenum=nil then           begin             if v>0 then               def.has_jumps:=true;             def.setmin(v);             def.setmax(v);           end         else           begin             { check for jumps }             if v>def.max+1 then              def.has_jumps:=true;             { update low and high }             if def.min>v then               def.setmin(v);             if def.max<v then               def.setmax(v);           end;         order;      end;    constructor tenumsym.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(enumsym,ppufile);         ppufile.getderef(definitionderef);         value:=ppufile.getlongint;         nextenum := Nil;      end;    procedure tenumsym.buildderef;      begin         definitionderef.build(definition);      end;    procedure tenumsym.deref;      begin         definition:=tenumdef(definitionderef.resolve);         order;      end;   procedure tenumsym.order;      var         sym : tenumsym;      begin         sym := tenumsym(definition.firstenum);         if sym = nil then          begin            definition.firstenum := self;            nextenum := nil;            exit;          end;         { reorder the symbols in increasing value }         if value < sym.value then          begin            nextenum := sym;            definition.firstenum := self;          end         else          begin            while (sym.value <= value) and assigned(sym.nextenum) do             sym := sym.nextenum;            nextenum := sym.nextenum;            sym.nextenum := self;          end;      end;    procedure tenumsym.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putderef(definitionderef);         ppufile.putlongint(value);         ppufile.writeentry(ibenumsym);      end;{****************************************************************************                                  TTYPESYM****************************************************************************}    constructor ttypesym.create(const n : string;def:tdef);      begin         inherited create(typesym,n);         typedef:=def;        { register the typesym for the definition }        if assigned(typedef) and           (typedef.typ<>errordef) and           not(assigned(typedef.typesym)) then         typedef.typesym:=self;      end;    constructor ttypesym.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(typesym,ppufile);         ppufile.getderef(typedefderef);      end;    procedure ttypesym.buildderef;      begin         typedefderef.build(typedef);      end;    procedure ttypesym.deref;      begin         typedef:=tdef(typedefderef.resolve);      end;    procedure ttypesym.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putderef(typedefderef);         ppufile.writeentry(ibtypesym);      end;{****************************************************************************                                  TSYSSYM****************************************************************************}    constructor tsyssym.create(const n : string;l : longint);      begin         inherited create(syssym,n);         number:=l;      end;    constructor tsyssym.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(syssym,ppufile);         number:=ppufile.getlongint;      end;    destructor tsyssym.destroy;      begin        inherited destroy;      end;    procedure tsyssym.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putlongint(number);         ppufile.writeentry(ibsyssym);      end;{*****************************************************************************                                 TMacro*****************************************************************************}    constructor tmacro.create(const n : string);      begin         inherited create(macrosym,n);         owner:=nil;         defined:=false;         is_used:=false;         is_compiler_var:=false;         buftext:=nil;         buflen:=0;      end;    constructor tmacro.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(macrosym,ppufile);         defined:=boolean(ppufile.getbyte);         is_compiler_var:=boolean(ppufile.getbyte);         is_used:=false;         buflen:= ppufile.getlongint;         if buflen > 0 then           begin             getmem(buftext, buflen);             ppufile.getdata(buftext^, buflen)           end         else           buftext:=nil;      end;    destructor tmacro.destroy;      begin         if assigned(buftext) then           freemem(buftext);         inherited destroy;      end;    procedure tmacro.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(defined));         ppufile.putbyte(byte(is_compiler_var));         ppufile.putlongint(buflen);         if buflen > 0 then           ppufile.putdata(buftext^,buflen);         ppufile.writeentry(ibmacrosym);      end;    function tmacro.GetCopy:tmacro;      var        p : tmacro;      begin        p:=tmacro.create(realname);        p.defined:=defined;        p.is_used:=is_used;        p.is_compiler_var:=is_compiler_var;        p.buflen:=buflen;        if assigned(buftext) then          begin            getmem(p.buftext,buflen);            move(buftext^,p.buftext^,buflen);          end;        Result:=p;      end;end.
 |