| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041 | {    Copyright (c) 1998-2002 by Florian Klaempfl, Pierre Muller    This unit handles the symbol tables    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 symtable;{$i fpcdefs.inc}interface    uses       { common }       cutils,cclasses,       { global }       cpuinfo,globtype,tokens,       { symtable }       symconst,symbase,symtype,symdef,symsym,       { ppu }       ppu,       { assembler }       aasmtai,aasmdata       ;{****************************************************************************                             Symtable types****************************************************************************}    type       tstoredsymtable = class(TSymtable)       private          b_needs_init_final : boolean;          procedure _needs_init_final(sym:TObject;arg:pointer);          procedure check_forward(sym:TObject;arg:pointer);          procedure labeldefined(sym:TObject;arg:pointer);          procedure varsymbolused(sym:TObject;arg:pointer);          procedure TestPrivate(sym:TObject;arg:pointer);          procedure objectprivatesymbolused(sym:TObject;arg:pointer);          procedure loaddefs(ppufile:tcompilerppufile);          procedure loadsyms(ppufile:tcompilerppufile);          procedure writedefs(ppufile:tcompilerppufile);          procedure writesyms(ppufile:tcompilerppufile);       public          procedure insert(sym:TSymEntry;checkdup:boolean=true);override;          procedure delete(sym:TSymEntry);override;          { load/write }          procedure ppuload(ppufile:tcompilerppufile);virtual;          procedure ppuwrite(ppufile:tcompilerppufile);virtual;          procedure buildderef;virtual;          procedure buildderefimpl;virtual;          procedure deref;virtual;          procedure derefimpl;virtual;          function  checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;override;          procedure allsymbolsused;          procedure allprivatesused;          procedure check_forwards;          procedure checklabels;          function  needs_init_final : boolean;          procedure testfordefaultproperty(sym:TObject;arg:pointer);       end;{$ifdef support_llvm}       tllvmshadowsymtableentry = class         constructor create(def: tdef; fieldoffset: aint);        private         ffieldoffset: aint;         fdef: tdef;        public         property fieldoffset: aint read ffieldoffset;         property def: tdef read fdef;       end;       tllvmshadowsymtable = class;{$endif support_llvm}              tabstractrecordsymtable = class(tstoredsymtable)       public          usefieldalignment,     { alignment to use for fields (PACKRECORDS value), C_alignment is C style }          recordalignment,       { alignment desired when inserting this record }          fieldalignment,        { alignment current alignment used when fields are inserted }          padalignment : shortint;   { size to a multiple of which the symtable has to be rounded up }{$ifdef support_llvm}          llvmst: tllvmshadowsymtable;{$endif}          constructor create(const n:string;usealign:shortint);          destructor destroy; override;          procedure ppuload(ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure alignrecord(fieldoffset:aint;varalign:shortint);          procedure addfield(sym:tfieldvarsym;vis:tvisibility);          procedure addalignmentpadding;          procedure insertdef(def:TDefEntry);override;          function is_packed: boolean;          function has_single_field(out sym:tfieldvarsym): boolean;        protected          _datasize       : aint;          { size in bits of the data in case of bitpacked record. Only important during construction, }          { no need to save in/restore from ppu file. datasize is always (databitsize+7) div 8.       }          databitsize    : aint;          procedure setdatasize(val: aint);        public          property datasize : aint read _datasize write setdatasize;       end;       trecordsymtable = class(tabstractrecordsymtable)       public          constructor create(usealign:shortint);          procedure insertunionst(unionst : trecordsymtable;offset : longint);       end;       tObjectSymtable = class(tabstractrecordsymtable)       public          constructor create(adefowner:tdef;const n:string;usealign:shortint);          function  checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;override;       end;{$ifdef support_llvm}       { llvm record definitions cannot contain variant/union parts, }       { you have to flatten them first. the tllvmshadowsymtable     }       { contains a flattened version of a record/object symtable    }       tllvmshadowsymtable = class        private         equivst: tabstractrecordsymtable;         curroffset: aint;        public         symdeflist: TFPObjectList;         constructor create(st: tabstractrecordsymtable);         destructor destroy; override;        private         // generate the table         procedure generate;         // helpers         procedure appenddefoffset(vardef:tdef; fieldoffset: aint; derefclass: boolean);         procedure findvariantstarts(variantstarts: tfplist);         procedure addalignmentpadding(finalsize: aint);         procedure buildmapping(variantstarts: tfplist);         procedure buildtable(variantstarts: tfplist);       end;{$endif support_llvm}       { tabstractlocalsymtable }       tabstractlocalsymtable = class(tstoredsymtable)       public          procedure ppuwrite(ppufile:tcompilerppufile);override;          function count_locals:longint;       end;       tlocalsymtable = class(tabstractlocalsymtable)       public          constructor create(adefowner:tdef;level:byte);          function  checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;override;       end;       tparasymtable = class(tabstractlocalsymtable)       public          constructor create(adefowner:tdef;level:byte);          function  checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;override;       end;       tabstractuniTSymtable = class(tstoredsymtable)       public          constructor create(const n : string;id:word);          function iscurrentunit:boolean;override;       end;       tglobalsymtable = class(tabstractuniTSymtable)       public          unittypecount : word;          constructor create(const n : string;id:word);          procedure ppuload(ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function  checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;override;       end;       tstaticsymtable = class(tabstractuniTSymtable)       public          constructor create(const n : string;id:word);          procedure ppuload(ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function  checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;override;       end;       twithsymtable = class(TSymtable)          withrefnode : tobject; { tnode }          constructor create(aowner:tdef;ASymList:TFPHashObjectList;refnode:tobject{tnode});          destructor  destroy;override;          procedure clear;override;          procedure insertdef(def:TDefEntry);override;        end;       tstt_excepTSymtable = class(TSymtable)       public          constructor create;       end;       tmacrosymtable = class(tstoredsymtable)       public          constructor create(exported: boolean);       end;       { tenumsymtable }       tenumsymtable = class(tstoredsymtable)       public          procedure insert(sym: TSymEntry; checkdup: boolean = true); override;          constructor create(adefowner:tdef);       end;    var       systemunit     : tglobalsymtable; { pointer to the system unit }{****************************************************************************                             Functions****************************************************************************}{*** Misc ***}    function  FullTypeName(def,otherdef:tdef):string;    procedure incompatibletypes(def1,def2:tdef);    procedure hidesym(sym:TSymEntry);    procedure duplicatesym(var hashedid:THashedIDString;dupsym,origsym:TSymEntry);{*** Search ***}    procedure addsymref(sym:tsym);    function  is_visible_for_object(symst:tsymtable;symvisibility:tvisibility;contextobjdef:tobjectdef):boolean;    function  is_visible_for_object(pd:tprocdef;contextobjdef:tobjectdef):boolean;    function  is_visible_for_object(sym:tsym;contextobjdef:tobjectdef):boolean;    function  searchsym(const s : TIDString;out srsym:tsym;out srsymtable:TSymtable):boolean;    function  searchsym_type(const s : TIDString;out srsym:tsym;out srsymtable:TSymtable):boolean;    function  searchsym_in_module(pm:pointer;const s : TIDString;out srsym:tsym;out srsymtable:TSymtable):boolean;    function  searchsym_in_named_module(const unitname, symname: TIDString; out srsym: tsym; out srsymtable: tsymtable): boolean;    function  searchsym_in_class(classh,contextclassh:tobjectdef;const s : TIDString;out srsym:tsym;out srsymtable:TSymtable):boolean;    function  searchsym_in_class_by_msgint(classh:tobjectdef;msgid:longint;out srdef : tdef;out srsym:tsym;out srsymtable:TSymtable):boolean;    function  searchsym_in_class_by_msgstr(classh:tobjectdef;const s:string;out srsym:tsym;out srsymtable:TSymtable):boolean;    function  search_system_type(const s: TIDString): ttypesym;    function  search_named_unit_globaltype(const unitname, typename: TIDString; throwerror: boolean): ttypesym;    function  search_class_member(pd : tobjectdef;const s : string):tsym;    function  search_assignment_operator(from_def,to_def:Tdef):Tprocdef;    function  search_enumerator_operator(type_def:Tdef):Tprocdef;    function  search_class_helper(pd : tobjectdef;const s : string; out srsym: tsym; out srsymtable: tsymtable):boolean;    function  search_objc_method(const s : string; out srsym: tsym; out srsymtable: tsymtable):boolean;    {Looks for macro s (must be given in upper case) in the macrosymbolstack, }    {and returns it if found. Returns nil otherwise.}    function  search_macro(const s : string):tsym;    { Additionally to searching for a macro, also checks whether it's still }    { actually defined (could be disable using "undef")                     }    function  defined_macro(const s : string):boolean;{*** Object Helpers ***}    function search_default_property(pd : tobjectdef) : tpropertysym;{*** Macro Helpers ***}    {If called initially, the following procedures manipulate macros in }    {initialmacrotable, otherwise they manipulate system macros local to a module.}    {Name can be given in any case (it will be converted to upper case).}    procedure def_system_macro(const name : string);    procedure set_system_macro(const name, value : string);    procedure set_system_compvar(const name, value : string);    procedure undef_system_macro(const name : string);{*** symtable stack ***}{ $ifdef DEBUG    procedure test_symtablestack;    procedure list_symtablestack; $endif DEBUG}{$ifdef UNITALIASES}    type       punit_alias = ^tunit_alias;       tunit_alias = object(TNamedIndexItem)          newname : pshortstring;          constructor init(const n:string);          destructor  done;virtual;       end;    var       unitaliases : pdictionary;    procedure addunitalias(const n:string);    function getunitalias(const n:string):string;{$endif UNITALIASES}{*** Init / Done ***}    procedure IniTSymtable;    procedure DoneSymtable;    const       overloaded_names : array [NOTOKEN..last_overloaded] of string[16] =         ('error',          'plus','minus','star','slash','equal',          'greater','lower','greater_or_equal',          'lower_or_equal',          'sym_diff','starstar',          'as','is','in','or',          'and','div','mod','not','shl','shr','xor',          'assign','enumerator');implementation    uses      { global }      verbose,globals,      { target }      systems,      { symtable }      symutil,defcmp,defutil,      { module }      fmodule,      { codegen }      procinfo      ;    var      dupnr : longint; { unique number for duplicate symbols }{*****************************************************************************                             TStoredSymtable*****************************************************************************}    procedure tstoredsymtable.insert(sym:TSymEntry;checkdup:boolean=true);      begin        inherited insert(sym,checkdup);      end;    procedure tstoredsymtable.delete(sym:TSymEntry);      begin        inherited delete(sym);      end;    procedure tstoredsymtable.ppuload(ppufile:tcompilerppufile);      begin        { load definitions }        loaddefs(ppufile);        { load symbols }        loadsyms(ppufile);      end;    procedure tstoredsymtable.ppuwrite(ppufile:tcompilerppufile);      begin         { write definitions }         writedefs(ppufile);         { write symbols }         writesyms(ppufile);      end;    procedure tstoredsymtable.loaddefs(ppufile:tcompilerppufile);      var        def : tdef;        b   : byte;      begin         { load start of definition section, which holds the amount of defs }         if ppufile.readentry<>ibstartdefs then           Message(unit_f_ppu_read_error);         { read definitions }         repeat           b:=ppufile.readentry;           case b of             ibpointerdef : def:=tpointerdef.ppuload(ppufile);             ibarraydef : def:=tarraydef.ppuload(ppufile);             iborddef : def:=torddef.ppuload(ppufile);             ibfloatdef : def:=tfloatdef.ppuload(ppufile);             ibprocdef : def:=tprocdef.ppuload(ppufile);             ibshortstringdef : def:=tstringdef.loadshort(ppufile);             iblongstringdef : def:=tstringdef.loadlong(ppufile);             ibansistringdef : def:=tstringdef.loadansi(ppufile);             ibwidestringdef : def:=tstringdef.loadwide(ppufile);             ibunicodestringdef : def:=tstringdef.loadunicode(ppufile);             ibrecorddef : def:=trecorddef.ppuload(ppufile);             ibobjectdef : def:=tobjectdef.ppuload(ppufile);             ibenumdef : def:=tenumdef.ppuload(ppufile);             ibsetdef : def:=tsetdef.ppuload(ppufile);             ibprocvardef : def:=tprocvardef.ppuload(ppufile);             ibfiledef : def:=tfiledef.ppuload(ppufile);             ibclassrefdef : def:=tclassrefdef.ppuload(ppufile);             ibformaldef : def:=tformaldef.ppuload(ppufile);             ibvariantdef : def:=tvariantdef.ppuload(ppufile);             ibundefineddef : def:=tundefineddef.ppuload(ppufile);             ibenddefs : break;             ibend : Message(unit_f_ppu_read_error);           else             Message1(unit_f_ppu_invalid_entry,tostr(b));           end;           InsertDef(def);         until false;      end;    procedure tstoredsymtable.loadsyms(ppufile:tcompilerppufile);      var        b   : byte;        sym : tsym;      begin      { load start of definition section, which holds the amount of defs }         if ppufile.readentry<>ibstartsyms then          Message(unit_f_ppu_read_error);         { now read the symbols }         repeat           b:=ppufile.readentry;           case b of                ibtypesym : sym:=ttypesym.ppuload(ppufile);                ibprocsym : sym:=tprocsym.ppuload(ppufile);               ibconstsym : sym:=tconstsym.ppuload(ppufile);           ibstaticvarsym : sym:=tstaticvarsym.ppuload(ppufile);            iblocalvarsym : sym:=tlocalvarsym.ppuload(ppufile);             ibparavarsym : sym:=tparavarsym.ppuload(ppufile);            ibfieldvarsym : sym:=tfieldvarsym.ppuload(ppufile);         ibabsolutevarsym : sym:=tabsolutevarsym.ppuload(ppufile);                ibenumsym : sym:=tenumsym.ppuload(ppufile);            ibpropertysym : sym:=tpropertysym.ppuload(ppufile);                ibunitsym : sym:=tunitsym.ppuload(ppufile);               iblabelsym : sym:=tlabelsym.ppuload(ppufile);                 ibsyssym : sym:=tsyssym.ppuload(ppufile);               ibmacrosym : sym:=tmacro.ppuload(ppufile);                ibendsyms : break;                    ibend : Message(unit_f_ppu_read_error);           else             Message1(unit_f_ppu_invalid_entry,tostr(b));           end;           Insert(sym,false);         until false;      end;    procedure tstoredsymtable.writedefs(ppufile:tcompilerppufile);      var        i   : longint;        def : tstoreddef;      begin        { each definition get a number, write then the amount of defs to the          ibstartdef entry }        ppufile.putlongint(DefList.count);        ppufile.writeentry(ibstartdefs);        { now write the definition }        for i:=0 to DefList.Count-1 do          begin            def:=tstoreddef(DefList[i]);            def.ppuwrite(ppufile);          end;        { write end of definitions }        ppufile.writeentry(ibenddefs);      end;    procedure tstoredsymtable.writesyms(ppufile:tcompilerppufile);      var        i   : longint;        sym : Tstoredsym;      begin        { each definition get a number, write then the amount of syms and the          datasize to the ibsymdef entry }        ppufile.putlongint(SymList.count);        ppufile.writeentry(ibstartsyms);        { foreach is used to write all symbols }        for i:=0 to SymList.Count-1 do          begin            sym:=tstoredsym(SymList[i]);            sym.ppuwrite(ppufile);          end;        { end of symbols }        ppufile.writeentry(ibendsyms);      end;    procedure tstoredsymtable.buildderef;      var        i   : longint;        def : tstoreddef;        sym : tstoredsym;      begin        { interface definitions }        for i:=0 to DefList.Count-1 do          begin            def:=tstoreddef(DefList[i]);            def.buildderef;          end;        { interface symbols }        for i:=0 to SymList.Count-1 do          begin            sym:=tstoredsym(SymList[i]);            sym.buildderef;          end;      end;    procedure tstoredsymtable.buildderefimpl;      var        i   : longint;        def : tstoreddef;      begin        { implementation definitions }        for i:=0 to DefList.Count-1 do          begin            def:=tstoreddef(DefList[i]);            def.buildderefimpl;          end;      end;    procedure tstoredsymtable.deref;      var        i   : longint;        def : tstoreddef;        sym : tstoredsym;      begin        { first deref the interface ttype symbols. This is needs          to be done before the interface defs are derefed, because          the interface defs can contain references to the type symbols          which then already need to contain a resolved typedef field (PFV) }        for i:=0 to SymList.Count-1 do          begin            sym:=tstoredsym(SymList[i]);            if sym.typ=typesym then              sym.deref;          end;        { interface definitions }        for i:=0 to DefList.Count-1 do          begin            def:=tstoreddef(DefList[i]);            def.deref;          end;        { interface symbols }        for i:=0 to SymList.Count-1 do          begin            sym:=tstoredsym(SymList[i]);            if sym.typ<>typesym then              sym.deref;          end;      end;    procedure tstoredsymtable.derefimpl;      var        i   : longint;        def : tstoreddef;      begin        { implementation definitions }        for i:=0 to DefList.Count-1 do          begin            def:=tstoreddef(DefList[i]);            def.derefimpl;          end;      end;    function tstoredsymtable.checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;      var        hsym : tsym;      begin        hsym:=tsym(FindWithHash(hashedid));        if assigned(hsym) then          DuplicateSym(hashedid,sym,hsym);        result:=assigned(hsym);      end;{**************************************             Callbacks**************************************}    procedure TStoredSymtable.check_forward(sym:TObject;arg:pointer);      begin         if tsym(sym).typ=procsym then           tprocsym(sym).check_forward         { check also object method table            }         { we needn't to test the def list          }         { because each object has to have a type sym,           only test objects declarations, not type renamings }         else          if (tsym(sym).typ=typesym) and             assigned(ttypesym(sym).typedef) and             (ttypesym(sym).typedef.typesym=ttypesym(sym)) and             (ttypesym(sym).typedef.typ=objectdef) then           tobjectdef(ttypesym(sym).typedef).check_forwards;      end;    procedure TStoredSymtable.labeldefined(sym:TObject;arg:pointer);      begin        if (tsym(sym).typ=labelsym) and           not(tlabelsym(sym).defined) then         begin           if tlabelsym(sym).used then            Message1(sym_e_label_used_and_not_defined,tlabelsym(sym).realname)           else            Message1(sym_w_label_not_defined,tlabelsym(sym).realname);         end;      end;    procedure TStoredSymtable.varsymbolused(sym:TObject;arg:pointer);      begin         if (tsym(sym).typ in [staticvarsym,localvarsym,paravarsym,fieldvarsym]) and            ((tsym(sym).owner.symtabletype in             [parasymtable,localsymtable,ObjectSymtable,staticsymtable])) then           begin            { unused symbol should be reported only if no }            { error is reported                     }            { if the symbol is in a register it is used   }            { also don't count the value parameters which have local copies }            { also don't claim for high param of open parameters (PM) }            if (Errorcount<>0) or               ([vo_is_hidden_para,vo_is_funcret] * tabstractvarsym(sym).varoptions = [vo_is_hidden_para]) or               (sp_internal in tsym(sym).symoptions) then              exit;            if (tstoredsym(sym).refs=0) then              begin                 if (vo_is_funcret in tabstractvarsym(sym).varoptions) then                   begin                     { don't warn about the result of constructors }                     if ((tsym(sym).owner.symtabletype<>localsymtable) or                        (tprocdef(tsym(sym).owner.defowner).proctypeoption<>potype_constructor)) and                        not(cs_opt_nodedfa in current_settings.optimizerswitches) then                       MessagePos(tsym(sym).fileinfo,sym_w_function_result_not_set)                   end                 else if (tsym(sym).owner.symtabletype=parasymtable) then                   MessagePos1(tsym(sym).fileinfo,sym_h_para_identifier_not_used,tsym(sym).prettyname)                 else if (tsym(sym).owner.symtabletype=ObjectSymtable) then                   MessagePos2(tsym(sym).fileinfo,sym_n_private_identifier_not_used,tobjectdef(tsym(sym).owner.defowner).RttiName,tsym(sym).prettyname)                 else                   MessagePos1(tsym(sym).fileinfo,sym_n_local_identifier_not_used,tsym(sym).prettyname);              end            else if tabstractvarsym(sym).varstate in [vs_written,vs_initialised] then              begin                 if (tsym(sym).owner.symtabletype=parasymtable) then                   begin                     if not(tabstractvarsym(sym).varspez in [vs_var,vs_out]) and                        not(vo_is_funcret in tabstractvarsym(sym).varoptions) then                       MessagePos1(tsym(sym).fileinfo,sym_h_para_identifier_only_set,tsym(sym).prettyname)                   end                 else if (tsym(sym).owner.symtabletype=ObjectSymtable) then                   MessagePos2(tsym(sym).fileinfo,sym_n_private_identifier_only_set,tobjectdef(tsym(sym).owner.defowner).RttiName,tsym(sym).prettyname)                 else if tabstractvarsym(sym).varoptions*[vo_is_funcret,vo_is_public,vo_is_external]=[] then                   MessagePos1(tsym(sym).fileinfo,sym_n_local_identifier_only_set,tsym(sym).prettyname);              end            else if (tabstractvarsym(sym).varstate = vs_read_not_warned) and                    ([vo_is_public,vo_is_external] * tabstractvarsym(sym).varoptions = []) then              MessagePos1(tsym(sym).fileinfo,sym_w_identifier_only_read,tsym(sym).prettyname)          end        else if ((tsym(sym).owner.symtabletype in              [ObjectSymtable,parasymtable,localsymtable,staticsymtable])) then          begin           if (Errorcount<>0) or              (sp_internal in tsym(sym).symoptions) then             exit;           { do not claim for inherited private fields !! }           if (tsym(sym).refs=0) and (tsym(sym).owner.symtabletype=ObjectSymtable) then             case tsym(sym).typ of               typesym:                 MessagePos2(tsym(sym).fileinfo,sym_n_private_type_not_used,tobjectdef(tsym(sym).owner.defowner).RttiName,tsym(sym).prettyname);               constsym:                 MessagePos2(tsym(sym).fileinfo,sym_n_private_const_not_used,tobjectdef(tsym(sym).owner.defowner).RttiName,tsym(sym).prettyname);               propertysym:                 MessagePos2(tsym(sym).fileinfo,sym_n_private_property_not_used,tobjectdef(tsym(sym).owner.defowner).RttiName,tsym(sym).prettyname);             else               MessagePos2(tsym(sym).fileinfo,sym_n_private_method_not_used,tobjectdef(tsym(sym).owner.defowner).RttiName,tsym(sym).prettyname);             end           { units references are problematic }           else            begin              if (tsym(sym).refs=0) and                 not(tsym(sym).typ in [enumsym,unitsym]) and                 not(is_funcret_sym(tsym(sym))) and                 { don't complain about compiler generated syms for specializations, see also #13405 }                 not((tsym(sym).typ=typesym) and (df_specialization in ttypesym(sym).typedef.defoptions) and                    (pos('$',ttypesym(sym).Realname)<>0)) and                 (                  (tsym(sym).typ<>procsym) or                  ((tsym(sym).owner.symtabletype=staticsymtable) and                   not current_module.is_unit)                 ) and                 { don't complain about alias for hidden _cmd parameter to                   obj-c methods }                 not((tsym(sym).typ in [localvarsym,paravarsym,absolutevarsym]) and                     (vo_is_msgsel in tabstractvarsym(sym).varoptions)) then                MessagePos2(tsym(sym).fileinfo,sym_h_local_symbol_not_used,SymTypeName[tsym(sym).typ],tsym(sym).prettyname);            end;          end;      end;    procedure TStoredSymtable.TestPrivate(sym:TObject;arg:pointer);      begin        if tsym(sym).visibility in [vis_private,vis_strictprivate] then          varsymbolused(sym,arg);      end;    procedure TStoredSymtable.objectprivatesymbolused(sym:TObject;arg:pointer);      begin         {           Don't test simple object aliases PM         }         if (tsym(sym).typ=typesym) and            (ttypesym(sym).typedef.typ=objectdef) and            (ttypesym(sym).typedef.typesym=tsym(sym)) then           tobjectdef(ttypesym(sym).typedef).symtable.SymList.ForEachCall(@TestPrivate,nil);      end;   procedure tstoredsymtable.testfordefaultproperty(sym:TObject;arg:pointer);     begin        if (tsym(sym).typ=propertysym) and           (ppo_defaultproperty in tpropertysym(sym).propoptions) then          ppointer(arg)^:=sym;     end;{***********************************************           Process all entries***********************************************}    { checks, if all procsyms and methods are defined }    procedure tstoredsymtable.check_forwards;      begin         SymList.ForEachCall(@check_forward,nil);      end;    procedure tstoredsymtable.checklabels;      begin         SymList.ForEachCall(@labeldefined,nil);      end;    procedure tstoredsymtable.allsymbolsused;      begin         SymList.ForEachCall(@varsymbolused,nil);      end;    procedure tstoredsymtable.allprivatesused;      begin         SymList.ForEachCall(@objectprivatesymbolused,nil);      end;    procedure TStoredSymtable._needs_init_final(sym:TObject;arg:pointer);      begin         if b_needs_init_final then          exit;         case tsym(sym).typ of           fieldvarsym,           staticvarsym,           localvarsym,           paravarsym :             begin               if is_managed_type(tabstractvarsym(sym).vardef) then                 b_needs_init_final:=true;             end;         end;      end;    { returns true, if p contains data which needs init/final code }    function tstoredsymtable.needs_init_final : boolean;      begin         b_needs_init_final:=false;         SymList.ForEachCall(@_needs_init_final,nil);         needs_init_final:=b_needs_init_final;      end;{****************************************************************************                          TAbstractRecordSymtable****************************************************************************}    constructor tabstractrecordsymtable.create(const n:string;usealign:shortint);      begin        inherited create(n);        _datasize:=0;        databitsize:=0;        recordalignment:=1;        usefieldalignment:=usealign;        padalignment:=1;        { recordalign C_alignment means C record packing, that starts          with an alignment of 1 }        case usealign of          C_alignment,          bit_alignment:            fieldalignment:=1;          mac68k_alignment:            fieldalignment:=2;          else            fieldalignment:=usealign;        end;      end;    destructor tabstractrecordsymtable.destroy;      begin{$ifdef support_llvm}        llvmst.free;{$endif suppor_llvm}        inherited destroy;      end;            procedure tabstractrecordsymtable.ppuload(ppufile:tcompilerppufile);      begin        inherited ppuload(ppufile);      end;    procedure tabstractrecordsymtable.ppuwrite(ppufile:tcompilerppufile);      var        oldtyp : byte;      begin         oldtyp:=ppufile.entrytyp;         ppufile.entrytyp:=subentryid;         inherited ppuwrite(ppufile);         ppufile.entrytyp:=oldtyp;      end;    function field2recordalignment(fieldoffs, fieldalign: aint): aint;      begin        { optimal alignment of the record when declaring a variable of this }        { type is independent of the packrecords setting                    }        if (fieldoffs mod fieldalign) = 0 then          result:=fieldalign        else if (fieldalign >= 16) and                ((fieldoffs mod 16) = 0) and                ((fieldalign mod 16) = 0) then          result:=16        else if (fieldalign >= 8) and                ((fieldoffs mod 8) = 0) and                ((fieldalign mod 8) = 0) then          result:=8        else if (fieldalign >= 4) and                ((fieldoffs mod 4) = 0) and                ((fieldalign mod 4) = 0) then          result:=4        else if (fieldalign >= 2) and                ((fieldoffs mod 2) = 0) and                ((fieldalign mod 2) = 0) then          result:=2        else          result:=1;      end;    procedure tabstractrecordsymtable.alignrecord(fieldoffset:aint;varalign:shortint);      var        varalignrecord: shortint;      begin        case usefieldalignment of          C_alignment:            varalignrecord:=used_align(varalign,current_settings.alignment.recordalignmin,current_settings.alignment.maxCrecordalign);          mac68k_alignment:            varalignrecord:=2;          else            varalignrecord:=field2recordalignment(fieldoffset,varalign);        end;        recordalignment:=max(recordalignment,varalignrecord);      end;    procedure tabstractrecordsymtable.addfield(sym:tfieldvarsym;vis:tvisibility);      var        l      : aint;        varalignfield,        varalign : shortint;        vardef : tdef;      begin        if (sym.owner<>self) then          internalerror(200602031);        if sym.fieldoffset<>-1 then          internalerror(200602032);        { set visibility for the symbol }        sym.visibility:=vis;        { this symbol can't be loaded to a register }        sym.varregable:=vr_none;        { Calculate field offset }        l:=sym.getsize;        vardef:=sym.vardef;        varalign:=vardef.alignment;        case usefieldalignment of          bit_alignment:            begin              { bitpacking only happens for ordinals, the rest is aligned at }              { 1 byte (compatible with GPC/GCC)                             }              if is_ordinal(vardef) then                begin                  sym.fieldoffset:=databitsize;                  l:=sym.getpackedbitsize;                end              else                begin                  databitsize:=_datasize*8;                  sym.fieldoffset:=databitsize;                  if (l>high(aint) div 8) then                    Message(sym_e_segment_too_large);                  l:=l*8;                end;              if varalign=0 then                varalign:=size_2_align(l);              recordalignment:=max(recordalignment,field2recordalignment(databitsize mod 8,varalign));              { bit packed records are limited to high(aint) bits }              { instead of bytes to avoid double precision        }              { arithmetic in offset calculations                 }              if int64(l)>high(aint)-sym.fieldoffset then                begin                  Message(sym_e_segment_too_large);                  _datasize:=high(aint);                  databitsize:=high(aint);                end              else                begin                  databitsize:=sym.fieldoffset+l;                  _datasize:=(databitsize+7) div 8;                end;              { rest is not applicable }              exit;            end;          { Calc the alignment size for C style records }          C_alignment:            begin              if (varalign>4) and                ((varalign mod 4)<>0) and                (vardef.typ=arraydef) then                Message1(sym_w_wrong_C_pack,vardef.typename);              if varalign=0 then                varalign:=l;              if (fieldalignment<current_settings.alignment.maxCrecordalign) then                begin                  if (varalign>16) and (fieldalignment<32) then                    fieldalignment:=32                  else if (varalign>12) and (fieldalignment<16) then                    fieldalignment:=16                  { 12 is needed for long double }                  else if (varalign>8) and (fieldalignment<12) then                    fieldalignment:=12                  else if (varalign>4) and (fieldalignment<8) then                    fieldalignment:=8                  else if (varalign>2) and (fieldalignment<4) then                    fieldalignment:=4                  else if (varalign>1) and (fieldalignment<2) then                    fieldalignment:=2;                end;              fieldalignment:=min(fieldalignment,current_settings.alignment.maxCrecordalign);            end;          mac68k_alignment:            begin              { mac68k alignment (C description):                 * char is aligned to 1 byte                 * everything else (except vector) is aligned to 2 bytes                 * vector is aligned to 16 bytes              }              if l>1 then                fieldalignment:=2              else                fieldalignment:=1;              varalign:=2;            end;        end;        if varalign=0 then          varalign:=size_2_align(l);        varalignfield:=used_align(varalign,current_settings.alignment.recordalignmin,fieldalignment);        sym.fieldoffset:=align(_datasize,varalignfield);        if l>high(aint)-sym.fieldoffset then          begin            Message(sym_e_segment_too_large);            _datasize:=high(aint);          end        else          _datasize:=sym.fieldoffset+l;        { Calc alignment needed for this record }        alignrecord(sym.fieldoffset,varalign);      end;    procedure tabstractrecordsymtable.addalignmentpadding;      begin        { make the record size aligned correctly so it can be          used as elements in an array. For C records we          use the fieldalignment, because that is updated with the          used alignment. }        if (padalignment = 1) then          case usefieldalignment of            C_alignment:              padalignment:=fieldalignment;            { bitpacked }            bit_alignment:              padalignment:=1;            { mac68k: always round to multiple of 2 }            mac68k_alignment:              padalignment:=2;            { default/no packrecords specified }            0:              padalignment:=recordalignment            { specific packrecords setting -> use as upper limit }            else              padalignment:=min(recordalignment,usefieldalignment);          end;        _datasize:=align(_datasize,padalignment);      end;    procedure tabstractrecordsymtable.insertdef(def:TDefEntry);      begin        { Enums must also be available outside the record scope,          insert in the owner of this symtable }        if def.typ=enumdef then          defowner.owner.insertdef(def)        else          inherited insertdef(def);      end;    function tabstractrecordsymtable.is_packed: boolean;      begin        result:=usefieldalignment=bit_alignment;      end;    function tabstractrecordsymtable.has_single_field(out sym: tfieldvarsym): boolean;      var        i: longint;      begin        result:=false;        { If a record contains a union, it does not contain a "single          non-composite field" in the context of certain ABIs requiring          special treatment for such records }        if (defowner.typ=recorddef) and           trecorddef(defowner).isunion then          exit;        { a record/object can contain other things than fields }        for i:=0 to SymList.Count-1 do          begin            if tsym(symlist[i]).typ=fieldvarsym then              begin                if result then                  begin                    result:=false;                    exit;                  end;                result:=true;                sym:=tfieldvarsym(symlist[i])              end;          end;      end;    procedure tabstractrecordsymtable.setdatasize(val: aint);      begin        _datasize:=val;        if (usefieldalignment=bit_alignment) then          { can overflow in non bitpacked records }          databitsize:=val*8;      end;{****************************************************************************                              TRecordSymtable****************************************************************************}    constructor trecordsymtable.create(usealign:shortint);      begin        inherited create('',usealign);        symtabletype:=recordsymtable;      end;   { this procedure is reserved for inserting case variant into      a record symtable }    { the offset is the location of the start of the variant      and datasize and dataalignment corresponds to      the complete size (see code in pdecl unit) PM }    procedure trecordsymtable.insertunionst(unionst : trecordsymtable;offset : longint);      var        sym : tsym;        def : tdef;        i : integer;        varalignrecord,varalign,        storesize,storealign : aint;        bitsize: aint;      begin        storesize:=_datasize;        storealign:=fieldalignment;        _datasize:=offset;        if (usefieldalignment=bit_alignment) then          databitsize:=offset*8;        { We move the ownership of the defs and symbols to the new recordsymtable.          The old unionsymtable keeps the references, but doesn't own the          objects anymore }        unionst.DefList.OwnsObjects:=false;        unionst.SymList.OwnsObjects:=false;        { copy symbols }        for i:=0 to unionst.SymList.Count-1 do          begin            sym:=TSym(unionst.SymList[i]);            if sym.typ<>fieldvarsym then              internalerror(200601272);            if tfieldvarsym(sym).fieldoffset=0 then              include(tfieldvarsym(sym).varoptions,vo_is_first_field);            { add to this record symtable }//            unionst.SymList.List.List^[i].Data:=nil;            sym.ChangeOwner(self);            varalign:=tfieldvarsym(sym).vardef.alignment;            if varalign=0 then              varalign:=size_2_align(tfieldvarsym(sym).getsize);            { retrieve size }            if (usefieldalignment=bit_alignment) then              begin                { bit packed records are limited to high(aint) bits }                { instead of bytes to avoid double precision        }                { arithmetic in offset calculations                 }                if is_ordinal(tfieldvarsym(sym).vardef) then                  bitsize:=tfieldvarsym(sym).getpackedbitsize                else                  begin                    bitsize:=tfieldvarsym(sym).getsize;                    if (bitsize>high(aint) div 8) then                      Message(sym_e_segment_too_large);                    bitsize:=bitsize*8;                  end;                if bitsize>high(aint)-databitsize then                  begin                    Message(sym_e_segment_too_large);                    _datasize:=high(aint);                    databitsize:=high(aint);                  end                else                  begin                    databitsize:=tfieldvarsym(sym).fieldoffset+offset*8;                    _datasize:=(databitsize+7) div 8;                  end;                tfieldvarsym(sym).fieldoffset:=databitsize;              varalignrecord:=field2recordalignment(tfieldvarsym(sym).fieldoffset div 8,varalign);              end            else              begin                if tfieldvarsym(sym).getsize>high(aint)-_datasize then                  begin                    Message(sym_e_segment_too_large);                    _datasize:=high(aint);                  end                else                  _datasize:=tfieldvarsym(sym).fieldoffset+offset;                { update address }                tfieldvarsym(sym).fieldoffset:=_datasize;                varalignrecord:=field2recordalignment(tfieldvarsym(sym).fieldoffset,varalign);              end;            { update alignment of this record }            if (usefieldalignment<>C_alignment) and               (usefieldalignment<>mac68k_alignment) then              recordalignment:=max(recordalignment,varalignrecord);          end;        { update alignment for C records }        if (usefieldalignment=C_alignment) and           (usefieldalignment<>mac68k_alignment) then          recordalignment:=max(recordalignment,unionst.recordalignment);        { Register defs in the new record symtable }        for i:=0 to unionst.DefList.Count-1 do          begin            def:=TDef(unionst.DefList[i]);            def.ChangeOwner(self);          end;        _datasize:=storesize;        fieldalignment:=storealign;        { If a record contains a union, it does not contain a "single          non-composite field" in the context of certain ABIs requiring          special treatment for such records }        if defowner.typ=recorddef then          trecorddef(defowner).isunion:=true;      end;{****************************************************************************                              TObjectSymtable****************************************************************************}    constructor tObjectSymtable.create(adefowner:tdef;const n:string;usealign:shortint);      begin        inherited create(n,usealign);        symtabletype:=ObjectSymtable;        defowner:=adefowner;      end;    function tObjectSymtable.checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;      var         hsym : tsym;      begin         result:=false;         if not assigned(defowner) then           internalerror(200602061);         { procsym and propertysym have special code           to override values in inherited classes. For other           symbols check for duplicates }         if not(sym.typ in [procsym,propertysym]) then           begin              { but private ids can be reused }              hsym:=search_class_member(tobjectdef(defowner),hashedid.id);              if assigned(hsym) and                 (                  (                   not(m_delphi in current_settings.modeswitches) and                   is_visible_for_object(hsym,tobjectdef(defowner))                  ) or                  (                   { In Delphi, you can repeat members of a parent class. You can't }                   { do this for objects however, and you (obviouly) can't          }                   { declare two fields with the same name in a single class        }                   (m_delphi in current_settings.modeswitches) and                   (                    is_object(tdef(defowner)) or                    (hsym.owner = self)                   )                  )                 ) then                begin                  DuplicateSym(hashedid,sym,hsym);                  result:=true;                end;           end         else           begin             if not(m_duplicate_names in current_settings.modeswitches) then               result:=inherited checkduplicate(hashedid,sym);           end;      end;{$ifdef support_llvm}{****************************************************************************                              tLlvmShadowSymtableEntry****************************************************************************}    constructor tllvmshadowsymtableentry.create(def: tdef; fieldoffset: aint);      begin        fdef:=def;        ffieldoffset:=fieldoffset;      end;{****************************************************************************                              TLlvmShadowSymtable****************************************************************************}    constructor tllvmshadowsymtable.create(st: tabstractrecordsymtable);      begin        equivst:=st;        curroffset:=0;        symdeflist:=tfpobjectlist.create(true);        generate;      end;          destructor tllvmshadowsymtable.destroy;      begin        symdeflist.free;      end;    procedure tllvmshadowsymtable.appenddefoffset(vardef:tdef; fieldoffset: aint; derefclass: boolean);      var        sizectr,        tmpsize: aint;      begin        case equivst.usefieldalignment of          C_alignment:            { default for llvm, don't add explicit padding }            symdeflist.add(tllvmshadowsymtableentry.create(vardef,fieldoffset));          bit_alignment:            begin              { curoffset: bit address after the previous field.      }              { llvm has no special support for bitfields in records, }              { so we replace them with plain bytes.                  }              { as soon as a single bit of a byte is allocated, we    }              { allocate the byte in the llvm shadow record           }              if (fieldoffset>curroffset) then                curroffset:=align(curroffset,8);              { fields in bitpacked records always start either right }              { after the previous one, or at the next byte boundary. }              if (curroffset<>fieldoffset) then                internalerror(2008051002);              if is_ordinal(vardef) and                 (vardef.packedbitsize mod 8 <> 0) then                begin                  tmpsize:=vardef.packedbitsize;                  sizectr:=tmpsize+7;                  repeat                    symdeflist.add(tllvmshadowsymtableentry.create(u8inttype,fieldoffset+(tmpsize+7)-sizectr));                    dec(sizectr,8);                  until (sizectr<=0);                  inc(curroffset,tmpsize);                end              else                begin                  symdeflist.add(tllvmshadowsymtableentry.create(vardef,fieldoffset));                  if not(derefclass) then                    inc(curroffset,vardef.size*8)                  else                    inc(curroffset,tobjectsymtable(tobjectdef(vardef).symtable).datasize*8);               end;            end          else            begin              { curoffset: address right after the previous field }              while (fieldoffset>curroffset) do                begin                  symdeflist.add(tllvmshadowsymtableentry.create(s8inttype,curroffset));                  inc(curroffset);                end;              symdeflist.add(tllvmshadowsymtableentry.create(vardef,fieldoffset));              if not(derefclass) then                inc(curroffset,vardef.size)              else                inc(curroffset,tobjectsymtable(tobjectdef(vardef).symtable).datasize);            end        end      end;    procedure tllvmshadowsymtable.addalignmentpadding(finalsize: aint);      begin        case equivst.usefieldalignment of          { already correct in this case }          bit_alignment,          { handled by llvm }          C_alignment:            ;          else            begin              { add padding fields }              while (finalsize>curroffset) do                begin                  symdeflist.add(tllvmshadowsymtableentry.create(s8inttype,curroffset));                  inc(curroffset);                end;            end;        end;      end;    procedure tllvmshadowsymtable.findvariantstarts(variantstarts: tfplist);      var        sym: tfieldvarsym;        lastoffset: aint;        newalignment: aint;        i, j: longint;      begin        i:=0;        while (i<equivst.symlist.count) do          begin            if (tsym(equivst.symlist[i]).typ<>fieldvarsym) then              begin                inc(i);                continue;              end;            sym:=tfieldvarsym(equivst.symlist[i]);            { a "better" algorithm might be to use the largest }            { variant in case of (bit)packing, since then      }            { alignment doesn't matter                         }            if (vo_is_first_field in sym.varoptions) then              begin                { we assume that all fields are processed in order. }                if (variantstarts.count<>0) then                  lastoffset:=tfieldvarsym(variantstarts[variantstarts.count-1]).fieldoffset                else                  lastoffset:=-1;                { new variant at same level as last one: use if higher alignment }                if (lastoffset=sym.fieldoffset) then                  begin                    if (equivst.fieldalignment<>bit_alignment) then                      newalignment:=used_align(sym.vardef.alignment,current_settings.alignment.recordalignmin,equivst.fieldalignment)                    else                      newalignment:=1;                    if (newalignment>tfieldvarsym(variantstarts[variantstarts.count-1]).vardef.alignment) then                      variantstarts[variantstarts.count-1]:=sym;                  end                { variant at deeper level than last one -> add }                else if (lastoffset<sym.fieldoffset) then                  variantstarts.add(sym)                else                  begin                    { a variant at a less deep level, so backtrack }                    j:=variantstarts.count-2;                    while (j>=0) do                      begin                        if (tfieldvarsym(variantstarts[j]).fieldoffset=sym.fieldoffset) then                          break;                        dec(j);                      end;                    if (j<0) then                      internalerror(2008051003);                    { new variant has higher alignment? }                    if (equivst.fieldalignment<>bit_alignment) then                      newalignment:=used_align(sym.vardef.alignment,current_settings.alignment.recordalignmin,equivst.fieldalignment)                    else                      newalignment:=1;                    { yes, replace and remove previous nested variants }                    if (newalignment>tfieldvarsym(variantstarts[j]).vardef.alignment) then                      begin                        variantstarts[j]:=sym;                        variantstarts.count:=j+1;                      end                   { no, skip this variant }                    else                      begin                        inc(i);                        while (i<equivst.symlist.count) and                              ((tsym(equivst.symlist[i]).typ<>fieldvarsym) or                               (tfieldvarsym(equivst.symlist[i]).fieldoffset>sym.fieldoffset)) do                          inc(i);                        continue;                      end;                  end;              end;            inc(i);          end;      end;    procedure tllvmshadowsymtable.buildtable(variantstarts: tfplist);      var        lastvaroffsetprocessed: aint;        i, equivcount, varcount: longint;      begin        { if it's an object/class, the first entry is the parent (if there is one) }        if (equivst.symtabletype=objectsymtable) and           assigned(tobjectdef(equivst.defowner).childof) then          appenddefoffset(tobjectdef(equivst.defowner).childof,0,is_class_or_interface_or_dispinterface(tobjectdef(equivst.defowner).childof));        equivcount:=equivst.symlist.count;        varcount:=0;        i:=0;        lastvaroffsetprocessed:=-1;        while (i<equivcount) do          begin            if (tsym(equivst.symlist[i]).typ<>fieldvarsym) then              begin                inc(i);                continue;              end;            { start of a new variant? }            if (vo_is_first_field in tfieldvarsym(equivst.symlist[i]).varoptions) then              begin                { if we want to process the same variant offset twice, it means that we  }                { got to the end and are trying to process the next variant part -> stop }                if (tfieldvarsym(equivst.symlist[i]).fieldoffset<=lastvaroffsetprocessed) then                  break;                if (varcount>=variantstarts.count) then                  internalerror(2008051005);                { new variant part -> use the one with the biggest alignment }                i:=equivst.symlist.indexof(tobject(variantstarts[varcount]));                lastvaroffsetprocessed:=tfieldvarsym(equivst.symlist[i]).fieldoffset;                inc(varcount);                if (i<0) then                  internalerror(2008051004);              end;            appenddefoffset(tfieldvarsym(equivst.symlist[i]).vardef,tfieldvarsym(equivst.symlist[i]).fieldoffset,false);            inc(i);          end;        addalignmentpadding(equivst.datasize);      end;    procedure tllvmshadowsymtable.buildmapping(variantstarts: tfplist);      var        i, varcount: longint;        shadowindex: longint;        equivcount : longint;      begin        varcount:=0;        shadowindex:=0;        equivcount:=equivst.symlist.count;        i:=0;        while (i < equivcount) do          begin            if (tsym(equivst.symlist[i]).typ<>fieldvarsym) then              begin                inc(i);                continue;              end;            { start of a new variant? }            if (vo_is_first_field in tfieldvarsym(equivst.symlist[i]).varoptions) then              begin                { back up to a less deeply nested variant level? }                while (tfieldvarsym(equivst.symlist[i]).fieldoffset<tfieldvarsym(variantstarts[varcount]).fieldoffset) do                  dec(varcount);                { it's possible that some variants are more deeply nested than the                  one we recorded in the shadowsymtable (since we recorded the one                  with the biggest alignment, not necessarily the biggest one in size                }                if (tfieldvarsym(equivst.symlist[i]).fieldoffset>tfieldvarsym(variantstarts[varcount]).fieldoffset) then                  varcount:=variantstarts.count-1                else if (tfieldvarsym(equivst.symlist[i]).fieldoffset<>tfieldvarsym(variantstarts[varcount]).fieldoffset) then                  internalerror(2008051006);                { reset the shadowindex to the start of this variant. }                { in case the llvmfieldnr is not (yet) set for this   }                { field, shadowindex will simply be reset to zero and }                { we'll start searching from the start of the record  }                shadowindex:=tfieldvarsym(variantstarts[varcount]).llvmfieldnr;                if (varcount<pred(variantstarts.count)) then                  inc(varcount);              end;            { find the last shadowfield whose offset <= the current field's offset }            while (tllvmshadowsymtableentry(symdeflist[shadowindex]).fieldoffset<tfieldvarsym(equivst.symlist[i]).fieldoffset) and                  (shadowindex<symdeflist.count-1) and                  (tllvmshadowsymtableentry(symdeflist[shadowindex+1]).fieldoffset>=tfieldvarsym(equivst.symlist[i]).fieldoffset) do              inc(shadowindex);            { set the field number and potential offset from that field (in case }            { of overlapping variants)                                           }            tfieldvarsym(equivst.symlist[i]).llvmfieldnr:=shadowindex;            tfieldvarsym(equivst.symlist[i]).offsetfromllvmfield:=              tfieldvarsym(equivst.symlist[i]).fieldoffset-tllvmshadowsymtableentry(symdeflist[shadowindex]).fieldoffset;            inc(i);          end;      end;    procedure tllvmshadowsymtable.generate;      var        variantstarts: tfplist;      begin        variantstarts:=tfplist.create;        { first go through the entire record and }        { store the fieldvarsyms of the variants }        { with the highest alignment             }        findvariantstarts(variantstarts);        { now go through the regular fields and the selected variants, }        { and add them to the  llvm shadow record symtable             }        buildtable(variantstarts);                { finally map all original fields to the llvm definition }        buildmapping(variantstarts);        variantstarts.free;              end;{$endif support_llvm}{****************************************************************************                          TAbstractLocalSymtable****************************************************************************}   procedure tabstractlocalsymtable.ppuwrite(ppufile:tcompilerppufile);      var        oldtyp : byte;      begin         oldtyp:=ppufile.entrytyp;         ppufile.entrytyp:=subentryid;         { write definitions }         writedefs(ppufile);         { write symbols }         writesyms(ppufile);         ppufile.entrytyp:=oldtyp;      end;    function tabstractlocalsymtable.count_locals:longint;      var        i   : longint;        sym : tsym;      begin        result:=0;        for i:=0 to SymList.Count-1 do          begin            sym:=tsym(SymList[i]);            { Count only varsyms, but ignore the funcretsym }            if (tsym(sym).typ in [localvarsym,paravarsym]) and               (tsym(sym)<>current_procinfo.procdef.funcretsym) and               (not(vo_is_parentfp in tabstractvarsym(sym).varoptions) or                (tstoredsym(sym).refs>0)) then              inc(result);         end;      end;{****************************************************************************                              TLocalSymtable****************************************************************************}    constructor tlocalsymtable.create(adefowner:tdef;level:byte);      begin        inherited create('');        defowner:=adefowner;        symtabletype:=localsymtable;        symtablelevel:=level;      end;    function tlocalsymtable.checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;      var        hsym : tsym;      begin        if not assigned(defowner) or           (defowner.typ<>procdef) then          internalerror(200602042);        result:=false;        hsym:=tsym(FindWithHash(hashedid));        if assigned(hsym) then          begin            { a local and the function can have the same              name in TP and Delphi, but RESULT not }            if (m_duplicate_names in current_settings.modeswitches) and               (hsym.typ in [absolutevarsym,localvarsym]) and               (vo_is_funcret in tabstractvarsym(hsym).varoptions) and               not((m_result in current_settings.modeswitches) and                   (vo_is_result in tabstractvarsym(hsym).varoptions)) then              HideSym(hsym)            else              DuplicateSym(hashedid,sym,hsym);            result:=true;            exit;          end;        { check also parasymtable, this needs to be done here becuase          of the special situation with the funcret sym that needs to be          hidden for tp and delphi modes }        hsym:=tsym(tabstractprocdef(defowner).parast.FindWithHash(hashedid));        if assigned(hsym) then          begin            { a local and the function can have the same              name in TP and Delphi, but RESULT not }            if (m_duplicate_names in current_settings.modeswitches) and               (sym.typ in [absolutevarsym,localvarsym]) and               (vo_is_funcret in tabstractvarsym(sym).varoptions) and               not((m_result in current_settings.modeswitches) and                   (vo_is_result in tabstractvarsym(sym).varoptions)) then              Hidesym(sym)            else              DuplicateSym(hashedid,sym,hsym);            result:=true;            exit;          end;        { check ObjectSymtable, skip this for funcret sym because          that will always be positive because it has the same name          as the procsym }        if not is_funcret_sym(sym) and           (defowner.typ=procdef) and           assigned(tprocdef(defowner)._class) and           (tprocdef(defowner).owner.defowner=tprocdef(defowner)._class) and           (            not(m_delphi in current_settings.modeswitches) or            is_object(tprocdef(defowner)._class)           ) then          result:=tprocdef(defowner)._class.symtable.checkduplicate(hashedid,sym);      end;{****************************************************************************                              TParaSymtable****************************************************************************}    constructor tparasymtable.create(adefowner:tdef;level:byte);      begin        inherited create('');        defowner:=adefowner;        symtabletype:=parasymtable;        symtablelevel:=level;      end;    function tparasymtable.checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;      begin        result:=inherited checkduplicate(hashedid,sym);        if result then          exit;        if not(m_duplicate_names in current_settings.modeswitches) and           (defowner.typ=procdef) and           assigned(tprocdef(defowner)._class) and           (tprocdef(defowner).owner.defowner=tprocdef(defowner)._class) and           (            not(m_delphi in current_settings.modeswitches) or            is_object(tprocdef(defowner)._class)           ) then          result:=tprocdef(defowner)._class.symtable.checkduplicate(hashedid,sym);      end;{****************************************************************************                         TAbstractUniTSymtable****************************************************************************}    constructor tabstractuniTSymtable.create(const n : string;id:word);      begin        inherited create(n);        moduleid:=id;      end;    function tabstractuniTSymtable.iscurrentunit:boolean;      begin        result:=assigned(current_module) and                (                 (current_module.globalsymtable=self) or                 (current_module.localsymtable=self)                );      end;{****************************************************************************                              TStaticSymtable****************************************************************************}    constructor tstaticsymtable.create(const n : string;id:word);      begin        inherited create(n,id);        symtabletype:=staticsymtable;        symtablelevel:=main_program_level;      end;    procedure tstaticsymtable.ppuload(ppufile:tcompilerppufile);      begin        inherited ppuload(ppufile);        { now we can deref the syms and defs }        deref;      end;    procedure tstaticsymtable.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);      end;    function tstaticsymtable.checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;      var        hsym : tsym;      begin        result:=false;        hsym:=tsym(FindWithHash(hashedid));        if assigned(hsym) then          begin            { Delphi (contrary to TP) you can have a symbol with the same name as the              unit, the unit can then not be accessed anymore using              <unit>.<id>, so we can hide the symbol }            if (m_delphi in current_settings.modeswitches) and               (hsym.typ=symconst.unitsym) then              HideSym(hsym)            else              DuplicateSym(hashedid,sym,hsym);            result:=true;            exit;          end;        if (current_module.localsymtable=self) and           assigned(current_module.globalsymtable) then          result:=tglobalsymtable(current_module.globalsymtable).checkduplicate(hashedid,sym);      end;{****************************************************************************                              TGlobalSymtable****************************************************************************}    constructor tglobalsymtable.create(const n : string;id:word);      begin         inherited create(n,id);         symtabletype:=globalsymtable;         symtablelevel:=main_program_level;      end;    procedure tglobalsymtable.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(ppufile);         { now we can deref the syms and defs }         deref;      end;    procedure tglobalsymtable.ppuwrite(ppufile:tcompilerppufile);      begin        { write the symtable entries }        inherited ppuwrite(ppufile);      end;    function tglobalsymtable.checkduplicate(var hashedid:THashedIDString;sym:TSymEntry):boolean;      var        hsym : tsym;      begin        result:=false;        hsym:=tsym(FindWithHash(hashedid));        if assigned(hsym) then          begin            { Delphi (contrary to TP) you can have a symbol with the same name as the              unit, the unit can then not be accessed anymore using              <unit>.<id>, so we can hide the symbol }            if (m_delphi in current_settings.modeswitches) and               (hsym.typ=symconst.unitsym) then              HideSym(hsym)            else              DuplicateSym(hashedid,sym,hsym);            result:=true;            exit;          end;      end;{****************************************************************************                              TWITHSYMTABLE****************************************************************************}    constructor twithsymtable.create(aowner:tdef;ASymList:TFPHashObjectList;refnode:tobject{tnode});      begin         inherited create('');         symtabletype:=withsymtable;         withrefnode:=refnode;         { Replace SymList with the passed symlist }         SymList.free;         SymList:=ASymList;         defowner:=aowner;      end;    destructor twithsymtable.destroy;      begin        withrefnode.free;        { Disable SymList because we don't Own it }        SymList:=nil;        inherited destroy;      end;    procedure twithsymtable.clear;      begin         { remove no entry from a withsymtable as it is only a pointer to the           recorddef  or objectdef symtable }      end;    procedure twithsymtable.insertdef(def:TDefEntry);      begin        { Definitions can't be registered in the withsymtable          because the withsymtable is removed after the with block.          We can't easily solve it here because the next symtable in the          stack is not known. }        internalerror(200602046);      end;{****************************************************************************                          TSTT_ExceptionSymtable****************************************************************************}    constructor tstt_excepTSymtable.create;      begin        inherited create('');        symtabletype:=stt_excepTSymtable;      end;{****************************************************************************                          TMacroSymtable****************************************************************************}    constructor tmacrosymtable.create(exported: boolean);      begin        inherited create('');        if exported then          symtabletype:=exportedmacrosymtable        else          symtabletype:=localmacrosymtable;        symtablelevel:=main_program_level;      end;{****************************************************************************                          TEnumSymtable****************************************************************************}    procedure tenumsymtable.insert(sym: TSymEntry; checkdup: boolean);      var        value: longint;        def: tenumdef;      begin        // defowner = nil only when we are loading from ppu        if defowner<>nil then          begin            { First entry? Then we need to set the minval }            value:=tenumsym(sym).value;            def:=tenumdef(defowner);            if SymList.count=0 then              begin                if value>0 then                  def.has_jumps:=true;                def.setmin(value);                def.setmax(value);              end            else              begin                { check for jumps }                if value>def.max+1 then                  def.has_jumps:=true;                { update low and high }                if def.min>value then                  def.setmin(value);                if def.max<value then                  def.setmax(value);              end;          end;        inherited insert(sym, checkdup);      end;    constructor tenumsymtable.create(adefowner: tdef);      begin        inherited Create('');        symtabletype:=enumsymtable;        defowner:=adefowner;      end;{*****************************************************************************                             Helper Routines*****************************************************************************}    function FullTypeName(def,otherdef:tdef):string;      var        s1,s2 : string;      begin        if def.typ=objectdef then          s1:=tobjectdef(def).RttiName        else          s1:=def.typename;        { When the names are the same try to include the unit name }        if assigned(otherdef) and           (def.owner.symtabletype in [globalsymtable,staticsymtable]) then          begin            s2:=otherdef.typename;            if upper(s1)=upper(s2) then              s1:=def.owner.realname^+'.'+s1;          end;        FullTypeName:=s1;      end;    procedure incompatibletypes(def1,def2:tdef);      begin        { When there is an errordef there is already an error message show }        if (def2.typ=errordef) or           (def1.typ=errordef) then          exit;        CGMessage2(type_e_incompatible_types,FullTypeName(def1,def2),FullTypeName(def2,def1));      end;    procedure hidesym(sym:TSymEntry);      begin        sym.realname:='$hidden'+sym.realname;        tsym(sym).visibility:=vis_hidden;      end;    procedure duplicatesym(var hashedid:THashedIDString;dupsym,origsym:TSymEntry);      var        st : TSymtable;      begin        Message1(sym_e_duplicate_id,tsym(origsym).realname);        { Write hint where the original symbol was found }        st:=finduniTSymtable(origsym.owner);        with tsym(origsym).fileinfo do          begin            if assigned(st) and               (st.symtabletype=globalsymtable) and               st.iscurrentunit then              Message2(sym_h_duplicate_id_where,current_module.sourcefiles.get_file_name(fileindex),tostr(line))            else if assigned(st.name) then              Message2(sym_h_duplicate_id_where,'unit '+st.name^,tostr(line));          end;        { Rename duplicate sym to an unreachable name, but it can be          inserted in the symtable without errors }        inc(dupnr);        hashedid.id:='dup'+tostr(dupnr)+hashedid.id;        if assigned(dupsym) then          include(tsym(dupsym).symoptions,sp_implicitrename);      end;{*****************************************************************************                                  Search*****************************************************************************}     procedure addsymref(sym:tsym);       begin         { symbol uses count }         sym.IncRefCount;         { unit uses count }         if assigned(current_module) and            (sym.owner.symtabletype=globalsymtable) then             begin               if tglobalsymtable(sym.owner).moduleid>=current_module.unitmapsize then                 internalerror(200501152);               inc(current_module.unitmap[tglobalsymtable(sym.owner).moduleid].refs);             end;       end;    function is_visible_for_object(symst:tsymtable;symvisibility:tvisibility;contextobjdef:tobjectdef):boolean;      var        symownerdef : tobjectdef;      begin        result:=false;        { Get objdectdef owner of the symtable for the is_related checks }        if not assigned(symst) or           (symst.symtabletype<>objectsymtable) then          internalerror(200810285);        symownerdef:=tobjectdef(symst.defowner);        case symvisibility of          vis_private :            begin              { private symbols are allowed when we are in the same                module as they are defined }              result:=(                       (symownerdef.owner.symtabletype in [globalsymtable,staticsymtable]) and                       (symownerdef.owner.iscurrentunit)                      ) or                      ( // the case of specialize inside the generic declaration                       (symownerdef.owner.symtabletype = objectsymtable) and                       (                         assigned(current_objectdef) and                         (                           (current_objectdef=symownerdef) or                           (current_objectdef.owner.moduleid=symownerdef.owner.moduleid)                         )                       ) or                       (                         not assigned(current_objectdef) and                         (symownerdef.owner.moduleid=current_module.moduleid)                       )                      );            end;          vis_strictprivate :            begin              result:=assigned(current_objectdef) and                      (current_objectdef=symownerdef);            end;          vis_strictprotected :            begin               result:=assigned(current_objectdef) and                       current_objectdef.is_related(symownerdef);            end;          vis_protected :            begin              { protected symbols are visible in the module that defines them and                also visible to related objects. The related object must be defined                in the current module }              result:=(                       (                        (symownerdef.owner.symtabletype in [globalsymtable,staticsymtable]) and                        (symownerdef.owner.iscurrentunit)                       ) or                       (                        assigned(contextobjdef) and                        (contextobjdef.owner.symtabletype in [globalsymtable,staticsymtable]) and                        (contextobjdef.owner.iscurrentunit) and                        contextobjdef.is_related(symownerdef)                       ) or                       ( // the case of specialize inside the generic declaration                        (symownerdef.owner.symtabletype = objectsymtable) and                        (                          assigned(current_objectdef) and                          (                            (current_objectdef=symownerdef) or                            (current_objectdef.owner.moduleid=symownerdef.owner.moduleid)                          )                        ) or                        (                          not assigned(current_objectdef) and                          (symownerdef.owner.moduleid=current_module.moduleid)                         )                       )                      );            end;          vis_public,          vis_published :            result:=true;        end;      end;    function is_visible_for_object(pd:tprocdef;contextobjdef:tobjectdef):boolean;      begin        result:=is_visible_for_object(pd.owner,pd.visibility,contextobjdef);      end;    function is_visible_for_object(sym:tsym;contextobjdef:tobjectdef):boolean;      var        i  : longint;        pd : tprocdef;      begin        if sym.typ=procsym then          begin            { A procsym is visible, when there is at least one of the procdefs visible }            result:=false;            for i:=0 to tprocsym(sym).ProcdefList.Count-1 do              begin                pd:=tprocdef(tprocsym(sym).ProcdefList[i]);                if (pd.owner=sym.owner) and                   is_visible_for_object(pd,contextobjdef) then                  begin                    result:=true;                    exit;                  end;              end;          end        else          result:=is_visible_for_object(sym.owner,sym.visibility,contextobjdef);      end;    function  searchsym(const s : TIDString;out srsym:tsym;out srsymtable:TSymtable):boolean;      var        hashedid   : THashedIDString;        contextobjdef : tobjectdef;        stackitem  : psymtablestackitem;      begin        result:=false;        hashedid.id:=s;        stackitem:=symtablestack.stack;        while assigned(stackitem) do          begin            srsymtable:=stackitem^.symtable;            srsym:=tsym(srsymtable.FindWithHash(hashedid));            if assigned(srsym) then              begin                { use the class from withsymtable only when it is                  defined in this unit }                if (srsymtable.symtabletype=withsymtable) and                   assigned(srsymtable.defowner) and                   (srsymtable.defowner.typ=objectdef) and                   (srsymtable.defowner.owner.symtabletype in [globalsymtable,staticsymtable]) and                   (srsymtable.defowner.owner.iscurrentunit) then                  contextobjdef:=tobjectdef(srsymtable.defowner)                else                  contextobjdef:=current_objectdef;                if (srsym.owner.symtabletype<>objectsymtable) or                   is_visible_for_object(srsym,contextobjdef) then                  begin                    { we need to know if a procedure references symbols                      in the static symtable, because then it can't be                      inlined from outside this unit }                    if assigned(current_procinfo) and                       (srsym.owner.symtabletype=staticsymtable) then                      include(current_procinfo.flags,pi_uses_static_symtable);                    addsymref(srsym);                    result:=true;                    exit;                  end;              end;            { also search for class helpers }            if (srsymtable.symtabletype=objectsymtable) and               is_objcclass(tdef(srsymtable.defowner)) then              begin                if search_class_helper(tobjectdef(srsymtable.defowner),s,srsym,srsymtable) then                  begin                    result:=true;                    exit;                  end;              end;            stackitem:=stackitem^.next;          end;        srsym:=nil;        srsymtable:=nil;      end;    function searchsym_type(const s : TIDString;out srsym:tsym;out srsymtable:TSymtable):boolean;      var        hashedid  : THashedIDString;        stackitem : psymtablestackitem;      begin        result:=false;        hashedid.id:=s;        stackitem:=symtablestack.stack;        while assigned(stackitem) do          begin            {              It is not possible to have type symbols in:                records                objects                parameters              Exception are classes, objects, generic definitions and specializations              that have the parameterized types inserted in the symtable.            }            srsymtable:=stackitem^.symtable;            if not(srsymtable.symtabletype in [recordsymtable,ObjectSymtable,parasymtable]) or               (assigned(srsymtable.defowner) and                (                 (df_generic in tdef(srsymtable.defowner).defoptions) or                 (df_specialization in tdef(srsymtable.defowner).defoptions) or                 is_class_or_object(tdef(srsymtable.defowner)))                ) then              begin                srsym:=tsym(srsymtable.FindWithHash(hashedid));                if assigned(srsym) and                   not(srsym.typ in [fieldvarsym,paravarsym]) and                   (                    (srsym.owner.symtabletype<>objectsymtable) or                    (is_visible_for_object(srsym,current_objectdef) and                     (srsym.typ=typesym))                   ) then                  begin                    { we need to know if a procedure references symbols                      in the static symtable, because then it can't be                      inlined from outside this unit }                    if assigned(current_procinfo) and                       (srsym.owner.symtabletype=staticsymtable) then                      include(current_procinfo.flags,pi_uses_static_symtable);                    addsymref(srsym);                    result:=true;                    exit;                  end;              end;            stackitem:=stackitem^.next;          end;        result:=false;        srsym:=nil;        srsymtable:=nil;      end;    function searchsym_in_module(pm:pointer;const s : TIDString;out srsym:tsym;out srsymtable:TSymtable):boolean;      var        pmod : tmodule;      begin        pmod:=tmodule(pm);        result:=false;        if assigned(pmod.globalsymtable) then          begin            srsym:=tsym(pmod.globalsymtable.Find(s));            if assigned(srsym) then              begin                srsymtable:=pmod.globalsymtable;                addsymref(srsym);                result:=true;                exit;              end;          end;        { If the module is the current unit we also need          to search the local symtable }        if (pmod=current_module) and           assigned(pmod.localsymtable) then          begin            srsym:=tsym(pmod.localsymtable.Find(s));            if assigned(srsym) then              begin                srsymtable:=pmod.localsymtable;                addsymref(srsym);                result:=true;                exit;              end;          end;        srsym:=nil;        srsymtable:=nil;      end;    function searchsym_in_named_module(const unitname, symname: TIDString; out srsym: tsym; out srsymtable: tsymtable): boolean;      var        stackitem  : psymtablestackitem;      begin        result:=false;        stackitem:=symtablestack.stack;        while assigned(stackitem) do          begin            srsymtable:=stackitem^.symtable;            if (srsymtable.symtabletype=globalsymtable) and               (srsymtable.name^=unitname) then              begin                srsym:=tsym(srsymtable.find(symname));                if not assigned(srsym) then                  break;                result:=true;                exit;              end;            stackitem:=stackitem^.next;          end;        { If the module is the current unit we also need          to search the local symtable }        if (current_module.globalsymtable=srsymtable) and           assigned(current_module.localsymtable) then          begin            srsymtable:=current_module.localsymtable;            srsym:=tsym(srsymtable.find(symname));            if assigned(srsym) then              begin                result:=true;                exit;              end;          end;      end;    function find_real_objcclass_definition(pd: tobjectdef): tobjectdef;      var        hashedid   : THashedIDString;        stackitem  : psymtablestackitem;        srsymtable : tsymtable;        srsym      : tsym;      begin        hashedid.id:=pd.typesym.name;        stackitem:=symtablestack.stack;        while assigned(stackitem) do          begin            srsymtable:=stackitem^.symtable;            { ObjC classes can't appear in generics or as nested class              definitions }            if not(srsymtable.symtabletype in [recordsymtable,ObjectSymtable,parasymtable]) then              begin                srsym:=tsym(srsymtable.FindWithHash(hashedid));                if assigned(srsym) and                   (srsym.typ=typesym) and                   is_objcclass(ttypesym(srsym).typedef) and                   not(oo_is_formal in tobjectdef(ttypesym(srsym).typedef).objectoptions) then                  begin                    result:=tobjectdef(ttypesym(srsym).typedef);                    if assigned(current_procinfo) and                       (srsym.owner.symtabletype=staticsymtable) then                      include(current_procinfo.flags,pi_uses_static_symtable);                    addsymref(srsym);                    exit;                  end;              end;            stackitem:=stackitem^.next;          end;        { nothing found: give an error and return the original (empty) one }        Message1(sym_e_objc_formal_class_not_resolved,pd.objrealname^);        result:=pd;      end;    function searchsym_in_class(classh,contextclassh:tobjectdef;const s : TIDString;out srsym:tsym;out srsymtable:TSymtable):boolean;      var        hashedid : THashedIDString;        orgclass : tobjectdef;        i        : longint;      begin        orgclass:=classh;        { in case this is a formal objcclass, first find the real definition }        if assigned(classh) then          begin            if (oo_is_formal in classh.objectoptions) then              classh:=find_real_objcclass_definition(classh);            { The contextclassh is used for visibility. The classh must be equal to              or be a parent of contextclassh. E.g. for inherited searches the classh is the              parent. }            if not contextclassh.is_related(classh) then              internalerror(200811161);          end;        result:=false;        hashedid.id:=s;        { an Objective-C protocol can inherit from multiple other protocols          -> uses ImplementedInterfaces instead }        if is_objcprotocol(classh) then          begin            srsymtable:=classh.symtable;            srsym:=tsym(srsymtable.FindWithHash(hashedid));            if assigned(srsym) and               is_visible_for_object(srsym,contextclassh) then              begin                addsymref(srsym);                result:=true;                exit;              end;            for i:=0 to classh.ImplementedInterfaces.count-1 do              begin                if searchsym_in_class(TImplementedInterface(classh.ImplementedInterfaces[i]).intfdef,contextclassh,s,srsym,srsymtable) then                  begin                    result:=true;                    exit;                  end;              end;          end        else          begin            while assigned(classh) do              begin                srsymtable:=classh.symtable;                srsym:=tsym(srsymtable.FindWithHash(hashedid));                if assigned(srsym) and                   is_visible_for_object(srsym,contextclassh) then                  begin                    addsymref(srsym);                    result:=true;                    exit;                  end;                classh:=classh.childof;              end;          end;        if is_objcclass(orgclass) then          result:=search_class_helper(orgclass,s,srsym,srsymtable)        else          begin            srsym:=nil;            srsymtable:=nil;          end;      end;    function searchsym_in_class_by_msgint(classh:tobjectdef;msgid:longint;out srdef : tdef;out srsym:tsym;out srsymtable:TSymtable):boolean;      var        def : tdef;        i   : longint;      begin        { in case this is a formal objcclass, first find the real definition }        if assigned(classh) and           (oo_is_formal in classh.objectoptions) then          classh:=find_real_objcclass_definition(classh);        result:=false;        def:=nil;        while assigned(classh) do          begin            for i:=0 to classh.symtable.DefList.Count-1 do              begin                def:=tstoreddef(classh.symtable.DefList[i]);                { Find also all hidden private methods to                  be compatible with delphi, see tw6203 (PFV) }                if (def.typ=procdef) and                   (po_msgint in tprocdef(def).procoptions) and                   (tprocdef(def).messageinf.i=msgid) then                  begin                    srdef:=def;                    srsym:=tprocdef(def).procsym;                    srsymtable:=classh.symtable;                    addsymref(srsym);                    result:=true;                    exit;                  end;              end;            classh:=classh.childof;          end;        srdef:=nil;        srsym:=nil;        srsymtable:=nil;      end;    function searchsym_in_class_by_msgstr(classh:tobjectdef;const s:string;out srsym:tsym;out srsymtable:TSymtable):boolean;      var        def : tdef;        i   : longint;      begin        { in case this is a formal objcclass, first find the real definition }        if assigned(classh) and           (oo_is_formal in classh.objectoptions) then          classh:=find_real_objcclass_definition(classh);        result:=false;        def:=nil;        while assigned(classh) do          begin            for i:=0 to classh.symtable.DefList.Count-1 do              begin                def:=tstoreddef(classh.symtable.DefList[i]);                { Find also all hidden private methods to                  be compatible with delphi, see tw6203 (PFV) }                if (def.typ=procdef) and                   (po_msgstr in tprocdef(def).procoptions) and                   (tprocdef(def).messageinf.str^=s) then                  begin                    srsym:=tprocdef(def).procsym;                    srsymtable:=classh.symtable;                    addsymref(srsym);                    result:=true;                    exit;                  end;              end;            classh:=classh.childof;          end;        srsym:=nil;        srsymtable:=nil;      end;    function search_assignment_operator(from_def,to_def:Tdef):Tprocdef;      var        sym : Tprocsym;        hashedid : THashedIDString;        curreq,        besteq : tequaltype;        currpd,        bestpd : tprocdef;        stackitem : psymtablestackitem;      begin        hashedid.id:='assign';        besteq:=te_incompatible;        bestpd:=nil;        stackitem:=symtablestack.stack;        while assigned(stackitem) do          begin            sym:=Tprocsym(stackitem^.symtable.FindWithHash(hashedid));            if sym<>nil then              begin                if sym.typ<>procsym then                  internalerror(200402031);                { if the source type is an alias then this is only the second choice,                  if you mess with this code, check tw4093 }                currpd:=sym.find_procdef_assignment_operator(from_def,to_def,curreq);                if curreq>besteq then                  begin                    besteq:=curreq;                    bestpd:=currpd;                    if (besteq=te_exact) then                      break;                  end;              end;            stackitem:=stackitem^.next;          end;        result:=bestpd;      end;    function search_enumerator_operator(type_def:Tdef): Tprocdef;      var        sym : Tprocsym;        hashedid : THashedIDString;        curreq,        besteq : tequaltype;        currpd,        bestpd : tprocdef;        stackitem : psymtablestackitem;      begin        hashedid.id:='enumerator';        besteq:=te_incompatible;        bestpd:=nil;        stackitem:=symtablestack.stack;        while assigned(stackitem) do          begin            sym:=Tprocsym(stackitem^.symtable.FindWithHash(hashedid));            if sym<>nil then              begin                if sym.typ<>procsym then                  internalerror(200910241);                { if the source type is an alias then this is only the second choice,                  if you mess with this code, check tw4093 }                currpd:=sym.find_procdef_enumerator_operator(type_def,curreq);                if curreq>besteq then                  begin                    besteq:=curreq;                    bestpd:=currpd;                    if (besteq=te_exact) then                      break;                  end;              end;            stackitem:=stackitem^.next;          end;        result:=bestpd;    end;    function search_system_type(const s: TIDString): ttypesym;      var        sym : tsym;      begin        sym:=tsym(systemunit.Find(s));        if not assigned(sym) or           (sym.typ<>typesym) then          cgmessage1(cg_f_unknown_system_type,s);        result:=ttypesym(sym);      end;    function search_named_unit_globaltype(const unitname, typename: TIDString; throwerror: boolean): ttypesym;      var        srsymtable: tsymtable;        sym: tsym;      begin        if searchsym_in_named_module(unitname,typename,sym,srsymtable) and           (sym.typ=typesym) then          begin            result:=ttypesym(sym);            exit;          end        else          begin            if throwerror then              cgmessage2(cg_f_unknown_type_in_unit,typename,unitname);            result:=nil;          end;      end;    function search_class_helper(pd : tobjectdef;const s : string; out srsym: tsym; out srsymtable: tsymtable):boolean;      var        hashedid   : THashedIDString;        stackitem  : psymtablestackitem;        i          : longint;        defowner   : tobjectdef;      begin        hashedid.id:=class_helper_prefix+s;        stackitem:=symtablestack.stack;        while assigned(stackitem) do          begin            srsymtable:=stackitem^.symtable;            srsym:=tsym(srsymtable.FindWithHash(hashedid));            if assigned(srsym) then              begin                if not(srsymtable.symtabletype in [globalsymtable,staticsymtable]) or                   not(srsym.owner.symtabletype in [globalsymtable,staticsymtable]) or                   (srsym.typ<>procsym) then                  internalerror(2009111505);                { check whether this procsym includes a helper for this particular class }                for i:=0 to tprocsym(srsym).procdeflist.count-1 do                  begin                    { does pd inherit from (or is the same as) the class                      that this method's category extended?                      Warning: this list contains both category and objcclass methods                       (for id.randommethod), so only check category methods here                    }                    defowner:=tobjectdef(tprocdef(tprocsym(srsym).procdeflist[i]).owner.defowner);                    if (oo_is_classhelper in defowner.objectoptions) and                       pd.is_related(defowner.childof) then                      begin                        { we need to know if a procedure references symbols                          in the static symtable, because then it can't be                          inlined from outside this unit }                        if assigned(current_procinfo) and                           (srsym.owner.symtabletype=staticsymtable) then                          include(current_procinfo.flags,pi_uses_static_symtable);                        { no need to keep looking. There might be other                          categories that extend this, a parent or child                          class with a method with the same name (either                          overriding this one, or overridden by this one),                          but that doesn't matter as far as the basic                          procsym is concerned.                        }                        srsym:=tprocdef(tprocsym(srsym).procdeflist[i]).procsym;                        srsymtable:=srsym.owner;                        addsymref(srsym);                        result:=true;                        exit;                      end;                  end;              end;            stackitem:=stackitem^.next;          end;        srsym:=nil;        srsymtable:=nil;        result:=false;      end;    function search_objc_method(const s : string; out srsym: tsym; out srsymtable: tsymtable):boolean;      var        hashedid   : THashedIDString;        stackitem  : psymtablestackitem;        i          : longint;      begin        hashedid.id:=class_helper_prefix+s;        stackitem:=symtablestack.stack;        while assigned(stackitem) do          begin            srsymtable:=stackitem^.symtable;            srsym:=tsym(srsymtable.FindWithHash(hashedid));            if assigned(srsym) then              begin                if not(srsymtable.symtabletype in [globalsymtable,staticsymtable]) or                   not(srsym.owner.symtabletype in [globalsymtable,staticsymtable]) or                   (srsym.typ<>procsym) then                  internalerror(2009112005);                { check whether this procsym includes a helper for this particular class }                for i:=0 to tprocsym(srsym).procdeflist.count-1 do                  begin                    { we need to know if a procedure references symbols                      in the static symtable, because then it can't be                      inlined from outside this unit }                    if assigned(current_procinfo) and                       (srsym.owner.symtabletype=staticsymtable) then                      include(current_procinfo.flags,pi_uses_static_symtable);                    { no need to keep looking. There might be other                      methods with the same name, but that doesn't matter                      as far as the basic procsym is concerned.                    }                    srsym:=tprocdef(tprocsym(srsym).procdeflist[i]).procsym;                    { We need the symtable in which the classhelper-like sym                      is located, not the objectdef. The reason is that the                      callnode will climb the symtablestack until it encounters                      this symtable to start looking for overloads (and it won't                      find the objectsymtable in which this method sym is                      located                    srsymtable:=srsym.owner;                    }                    addsymref(srsym);                    result:=true;                    exit;                  end;              end;            stackitem:=stackitem^.next;          end;        srsym:=nil;        srsymtable:=nil;        result:=false;      end;    function search_class_member(pd : tobjectdef;const s : string):tsym;    { searches n in symtable of pd and all anchestors }      var        hashedid   : THashedIDString;        srsym      : tsym;        orgpd      : tobjectdef;        srsymtable : tsymtable;      begin        { in case this is a formal objcclass, first find the real definition }        if (oo_is_formal in pd.objectoptions) then          pd:=find_real_objcclass_definition(pd);        hashedid.id:=s;        orgpd:=pd;        while assigned(pd) do         begin           srsym:=tsym(pd.symtable.FindWithHash(hashedid));           if assigned(srsym) then            begin              search_class_member:=srsym;              exit;            end;           pd:=pd.childof;         end;        { not found, now look for class helpers }        if is_objcclass(pd) then          search_class_helper(orgpd,s,result,srsymtable)        else          result:=nil;      end;    function search_macro(const s : string):tsym;      var        stackitem  : psymtablestackitem;        hashedid   : THashedIDString;        srsym      : tsym;      begin        hashedid.id:=s;        { First search the localmacrosymtable before searching the          global macrosymtables from the units }        if assigned(current_module) then          begin            srsym:=tsym(current_module.localmacrosymtable.FindWithHash(hashedid));            if assigned(srsym) then              begin                result:= srsym;                exit;              end;          end;        stackitem:=macrosymtablestack.stack;        while assigned(stackitem) do          begin            srsym:=tsym(stackitem^.symtable.FindWithHash(hashedid));            if assigned(srsym) then              begin                result:= srsym;                exit;              end;            stackitem:=stackitem^.next;          end;        result:= nil;      end;    function defined_macro(const s : string):boolean;      var        mac: tmacro;      begin        mac:=tmacro(search_macro(s));        if assigned(mac) then          begin            mac.is_used:=true;            defined_macro:=mac.defined;          end        else          defined_macro:=false;      end;{****************************************************************************                              Object Helpers****************************************************************************}   function search_default_property(pd : tobjectdef) : tpropertysym;   { returns the default property of a class, searches also anchestors }     var       _defaultprop : tpropertysym;     begin        _defaultprop:=nil;        while assigned(pd) do          begin             pd.symtable.SymList.ForEachCall(@tstoredsymtable(pd.symtable).testfordefaultproperty,@_defaultprop);             if assigned(_defaultprop) then               break;             pd:=pd.childof;          end;        search_default_property:=_defaultprop;     end;{****************************************************************************                              Macro Helpers****************************************************************************}    procedure def_system_macro(const name : string);      var        mac : tmacro;        s: string;      begin         if name = '' then           internalerror(2004121202);         s:= upper(name);         mac:=tmacro(search_macro(s));         if not assigned(mac) then           begin             mac:=tmacro.create(s);             if assigned(current_module) then               current_module.localmacrosymtable.insert(mac)             else               initialmacrosymtable.insert(mac);           end;         if not mac.defined then           Message1(parser_c_macro_defined,mac.name);         mac.defined:=true;      end;    procedure set_system_macro(const name, value : string);      var        mac : tmacro;        s: string;      begin        if name = '' then          internalerror(2004121203);         s:= upper(name);         mac:=tmacro(search_macro(s));         if not assigned(mac) then           begin             mac:=tmacro.create(s);             if assigned(current_module) then               current_module.localmacrosymtable.insert(mac)             else               initialmacrosymtable.insert(mac);           end         else           begin             mac.is_compiler_var:=false;             if assigned(mac.buftext) then               freemem(mac.buftext,mac.buflen);           end;         Message2(parser_c_macro_set_to,mac.name,value);         mac.buflen:=length(value);         getmem(mac.buftext,mac.buflen);         move(value[1],mac.buftext^,mac.buflen);         mac.defined:=true;      end;    procedure set_system_compvar(const name, value : string);      var        mac : tmacro;        s: string;      begin        if name = '' then          internalerror(2004121204);         s:= upper(name);         mac:=tmacro(search_macro(s));         if not assigned(mac) then           begin             mac:=tmacro.create(s);             mac.is_compiler_var:=true;             if assigned(current_module) then               current_module.localmacrosymtable.insert(mac)             else               initialmacrosymtable.insert(mac);           end         else           begin             mac.is_compiler_var:=true;             if assigned(mac.buftext) then               freemem(mac.buftext,mac.buflen);           end;         Message2(parser_c_macro_set_to,mac.name,value);         mac.buflen:=length(value);         getmem(mac.buftext,mac.buflen);         move(value[1],mac.buftext^,mac.buflen);         mac.defined:=true;      end;    procedure undef_system_macro(const name : string);      var        mac : tmacro;        s: string;      begin         if name = '' then           internalerror(2004121205);         s:= upper(name);         mac:=tmacro(search_macro(s));         if not assigned(mac) then           {If not found, then it's already undefined.}         else           begin             if mac.defined then               Message1(parser_c_macro_undefined,mac.name);             mac.defined:=false;             mac.is_compiler_var:=false;             { delete old definition }             if assigned(mac.buftext) then               begin                  freemem(mac.buftext,mac.buflen);                  mac.buftext:=nil;               end;           end;      end;{$ifdef UNITALIASES}{****************************************************************************                              TUNIT_ALIAS ****************************************************************************}    constructor tunit_alias.create(const n:string);      var        i : longint;      begin        i:=pos('=',n);        if i=0 then         fail;        inherited createname(Copy(n,1,i-1));        newname:=stringdup(Copy(n,i+1,255));      end;    destructor tunit_alias.destroy;      begin        stringdispose(newname);        inherited destroy;      end;    procedure addunitalias(const n:string);      begin        unitaliases^.insert(tunit_alias,init(Upper(n))));      end;    function getunitalias(const n:string):string;      var        p : punit_alias;      begin        p:=punit_alias(unitaliases^.Find(Upper(n)));        if assigned(p) then         getunitalias:=punit_alias(p).newname^        else         getunitalias:=n;      end;{$endif UNITALIASES}{****************************************************************************                           Init/Done Symtable****************************************************************************}   procedure InitSymtable;     begin       { Reset symbolstack }       symtablestack:=nil;       systemunit:=nil;       { create error syms and def }       generrorsym:=terrorsym.create;       generrordef:=terrordef.create;       { macros }       initialmacrosymtable:=tmacrosymtable.create(false);       macrosymtablestack:=TSymtablestack.create;       macrosymtablestack.push(initialmacrosymtable);{$ifdef UNITALIASES}       { unit aliases }       unitaliases:=TFPHashObjectList.create;{$endif}       { set some global vars to nil, might be important for the ide }       class_tobject:=nil;       interface_iunknown:=nil;       rec_tguid:=nil;       dupnr:=0;     end;   procedure DoneSymtable;      begin        generrorsym.owner:=nil;        generrorsym.free;        generrordef.owner:=nil;        generrordef.free;        initialmacrosymtable.free;        macrosymtablestack.free;{$ifdef UNITALIASES}        unitaliases.free;{$endif}     end;end.
 |