| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563 | {    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       ;{****************************************************************************                             Symtable types****************************************************************************}    type       tstoredsymtable = class(tsymtable)       private          b_needs_init_final : boolean;          procedure _needs_init_final(p : tnamedindexitem;arg:pointer);          procedure check_forward(sym : TNamedIndexItem;arg:pointer);          procedure labeldefined(p : TNamedIndexItem;arg:pointer);          procedure varsymbolused(p : TNamedIndexItem;arg:pointer);          procedure TestPrivate(p : TNamedIndexItem;arg:pointer);          procedure objectprivatesymbolused(p : TNamedIndexItem;arg:pointer);          procedure unchain_overloads(p : TNamedIndexItem;arg:pointer);          procedure loaddefs(ppufile:tcompilerppufile);          procedure loadsyms(ppufile:tcompilerppufile);          procedure reset_def(def:Tnamedindexitem;arg:pointer);          procedure writedefs(ppufile:tcompilerppufile);          procedure writesyms(ppufile:tcompilerppufile);       public          { load/write }          procedure ppuload(ppufile:tcompilerppufile);virtual;          procedure ppuwrite(ppufile:tcompilerppufile);virtual;          procedure load_references(ppufile:tcompilerppufile;locals:boolean);virtual;          procedure write_references(ppufile:tcompilerppufile;locals:boolean);virtual;          procedure buildderef;virtual;          procedure buildderefimpl;virtual;          procedure deref;virtual;          procedure derefimpl;virtual;          procedure insert(sym : tsymentry);override;          procedure reset_all_defs;virtual;          function  speedsearch(const s : stringid;speedvalue : cardinal) : tsymentry;override;          procedure allsymbolsused;          procedure allprivatesused;          procedure check_forwards;          procedure checklabels;          function  needs_init_final : boolean;          procedure unchain_overloaded;{$ifdef GDB}          procedure concatstabto(asmlist : taasmoutput);virtual;          function  getnewtypecount : word; override;{$endif GDB}          procedure testfordefaultproperty(p : TNamedIndexItem;arg:pointer);       end;       tabstractrecordsymtable = class(tstoredsymtable)       public          datasize       : aint;          usefieldalignment,     { alignment to use for fields (PACKRECORDS value), -1 is C style }          recordalignment,       { alignment required 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 }          constructor create(const n:string;usealign:shortint);          procedure ppuload(ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure load_references(ppufile:tcompilerppufile;locals:boolean);override;          procedure write_references(ppufile:tcompilerppufile;locals:boolean);override;          procedure insertfield(sym:tfieldvarsym;addsym:boolean);          procedure derefimpl; override;          procedure addalignmentpadding;       end;       trecordsymtable = class(tabstractrecordsymtable)       public          constructor create(usealign:shortint);          procedure insertunionst(unionst : trecordsymtable;offset : longint);       end;       tobjectsymtable = class(tabstractrecordsymtable)       public          constructor create(const n:string;usealign:shortint);          procedure insert(sym : tsymentry);override;       end;       tabstractlocalsymtable = class(tstoredsymtable)       public          procedure ppuwrite(ppufile:tcompilerppufile);override;       end;       tlocalsymtable = class(tabstractlocalsymtable)       public          constructor create(level:byte);          procedure insert(sym : tsymentry);override;       end;       tparasymtable = class(tabstractlocalsymtable)       public          constructor create(level:byte);          procedure insert(sym : tsymentry);override;       end;       tabstractunitsymtable = class(tstoredsymtable)       public{$ifdef GDB}          dbx_count : longint;          prev_dbx_counter : plongint;          dbx_count_ok : boolean;{$endif GDB}          constructor create(const n : string;id:word);{$ifdef GDB}          procedure concattypestabto(asmlist : taasmoutput);{$endif GDB}          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;          procedure load_references(ppufile:tcompilerppufile;locals:boolean);override;          procedure write_references(ppufile:tcompilerppufile;locals:boolean);override;          procedure insert(sym : tsymentry);override;{$ifdef GDB}          function getnewtypecount : word; override;{$endif}       end;       tstaticsymtable = class(tabstractunitsymtable)       public          constructor create(const n : string;id:word);          procedure ppuload(ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure load_references(ppufile:tcompilerppufile;locals:boolean);override;          procedure write_references(ppufile:tcompilerppufile;locals:boolean);override;          procedure insert(sym : tsymentry);override;       end;       twithsymtable = class(tsymtable)          withrefnode : pointer; { tnode }          constructor create(aowner:tdef;asymsearch:TDictionary;refnode:pointer{tnode});          destructor  destroy;override;          procedure clear;override;        end;       tstt_exceptsymtable = class(tsymtable)       public          constructor create;       end;       tmacrosymtable = class(tstoredsymtable)       public          constructor create(exported: boolean);          procedure ppuload(ppufile:tcompilerppufile);override;       end;    var       constsymtable  : tsymtable;      { symtable were the constants can be inserted }       systemunit     : tglobalsymtable; { pointer to the system unit }{****************************************************************************                             Functions****************************************************************************}{*** Misc ***}    procedure globaldef(const s : string;var t:ttype);    function  findunitsymtable(st:tsymtable):tsymtable;    function  FullTypeName(def,otherdef:tdef):string;    procedure incompatibletypes(def1,def2:tdef);    procedure hidesym(sym:tsymentry);    procedure duplicatesym(dupsym,sym:tsymentry);{*** Search ***}    function  searchsym(const s : stringid;var srsym:tsym;var srsymtable:tsymtable):boolean;    function  searchsym_type(const s : stringid;var srsym:tsym;var srsymtable:tsymtable):boolean;    function  searchsymonlyin(p : tsymtable;const s : stringid):tsym;    function  searchsym_in_class(classh:tobjectdef;const s : stringid):tsym;    function  searchsym_in_class_by_msgint(classh:tobjectdef;i:longint):tsym;    function  searchsym_in_class_by_msgstr(classh:tobjectdef;const s:string):tsym;    function  searchsystype(const s: stringid; var srsym: ttypesym): boolean;{$ifdef notused}    function  searchsysvar(const s: stringid; var srsym: tsym; var symowner: tsymtable): boolean;{$endif notused}    function  search_class_member(pd : tobjectdef;const s : string):tsym;    function  search_assignment_operator(from_def,to_def:Tdef):Tprocdef;    {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;{*** Object Helpers ***}    procedure search_class_overloads(aprocsym : tprocsym);    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 : pstring;          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');implementation    uses      { global }      verbose,globals,      { target }      systems,      { symtable }      symutil,defcmp,      { module }      fmodule,{$ifdef GDB}      gdb,{$endif GDB}      { codegen }      procinfo      ;{*****************************************************************************                             TStoredSymtable*****************************************************************************}    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        hp : 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);         ppufile.getlongint;      { read definitions }         repeat           b:=ppufile.readentry;           case b of              ibpointerdef : hp:=tpointerdef.ppuload(ppufile);                ibarraydef : hp:=tarraydef.ppuload(ppufile);                  iborddef : hp:=torddef.ppuload(ppufile);                ibfloatdef : hp:=tfloatdef.ppuload(ppufile);                 ibprocdef : hp:=tprocdef.ppuload(ppufile);          ibshortstringdef : hp:=tstringdef.loadshort(ppufile);           iblongstringdef : hp:=tstringdef.loadlong(ppufile);{$ifdef ansistring_bits}         ibansistring16def : hp:=tstringdef.loadansi(ppufile,sb_16);         ibansistring32def : hp:=tstringdef.loadansi(ppufile,sb_32);         ibansistring64def : hp:=tstringdef.loadansi(ppufile,sb_64);{$else}           ibansistringdef : hp:=tstringdef.loadansi(ppufile);{$endif}           ibwidestringdef : hp:=tstringdef.loadwide(ppufile);               ibrecorddef : hp:=trecorddef.ppuload(ppufile);               ibobjectdef : hp:=tobjectdef.ppuload(ppufile);                 ibenumdef : hp:=tenumdef.ppuload(ppufile);                  ibsetdef : hp:=tsetdef.ppuload(ppufile);              ibprocvardef : hp:=tprocvardef.ppuload(ppufile);                 ibfiledef : hp:=tfiledef.ppuload(ppufile);             ibclassrefdef : hp:=tclassrefdef.ppuload(ppufile);               ibformaldef : hp:=tformaldef.ppuload(ppufile);              ibvariantdef : hp:=tvariantdef.ppuload(ppufile);                 ibenddefs : break;                     ibend : Message(unit_f_ppu_read_error);           else             Message1(unit_f_ppu_invalid_entry,tostr(b));           end;           hp.owner:=self;           defindex.insert(hp);         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);         { skip amount of symbols, not used currently }         ppufile.getlongint;         { 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);           ibglobalvarsym : sym:=tglobalvarsym.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);          ibtypedconstsym : sym:=ttypedconstsym.ppuload(ppufile);            ibpropertysym : sym:=tpropertysym.ppuload(ppufile);                ibunitsym : sym:=tunitsym.ppuload(ppufile);               iblabelsym : sym:=tlabelsym.ppuload(ppufile);                 ibsyssym : sym:=tsyssym.ppuload(ppufile);                ibrttisym : sym:=trttisym.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;           sym.owner:=self;           symindex.insert(sym);           symsearch.insert(sym);         until false;      end;    procedure tstoredsymtable.writedefs(ppufile:tcompilerppufile);      var         pd : tstoreddef;      begin         { each definition get a number, write then the amount of defs to the           ibstartdef entry }         ppufile.putlongint(defindex.count);         ppufile.writeentry(ibstartdefs);         { now write the definition }         pd:=tstoreddef(defindex.first);         while assigned(pd) do           begin              pd.ppuwrite(ppufile);              pd:=tstoreddef(pd.indexnext);           end;         { write end of definitions }         ppufile.writeentry(ibenddefs);      end;    procedure tstoredsymtable.writesyms(ppufile:tcompilerppufile);      var        pd : Tstoredsym;      begin         { each definition get a number, write then the amount of syms and the           datasize to the ibsymdef entry }         ppufile.putlongint(symindex.count);         ppufile.writeentry(ibstartsyms);         { foreach is used to write all symbols }         pd:=Tstoredsym(symindex.first);         while assigned(pd) do           begin              pd.ppuwrite(ppufile);              pd:=Tstoredsym(pd.indexnext);           end;         { end of symbols }         ppufile.writeentry(ibendsyms);      end;    procedure tstoredsymtable.load_references(ppufile:tcompilerppufile;locals:boolean);      var        b     : byte;        d     : tderef;        sym   : Tsym;        prdef : tstoreddef;      begin         b:=ppufile.readentry;         if b <> ibbeginsymtablebrowser then           Message1(unit_f_ppu_invalid_entry,tostr(b));         repeat           b:=ppufile.readentry;           case b of             ibsymref :               begin                 ppufile.getderef(d);                 sym:=Tsym(d.resolve);                 if assigned(sym) then                   sym.load_references(ppufile,locals);               end;             ibdefref :               begin                 ppufile.getderef(d);                 prdef:=tstoreddef(d.resolve);                 if assigned(prdef) then                   begin                     if prdef.deftype<>procdef then                       Message(unit_f_ppu_read_error);                     tprocdef(prdef).load_references(ppufile,locals);                   end;               end;             ibendsymtablebrowser :               break;             else               Message1(unit_f_ppu_invalid_entry,tostr(b));           end;         until false;      end;    procedure tstoredsymtable.write_references(ppufile:tcompilerppufile;locals:boolean);      var        pd : Tsym;      begin         ppufile.writeentry(ibbeginsymtablebrowser);         { write all symbols }         pd:=Tsym(symindex.first);         while assigned(pd) do           begin              pd.write_references(ppufile,locals);              pd:=Tsym(pd.indexnext);           end;         ppufile.writeentry(ibendsymtablebrowser);      end;    procedure tstoredsymtable.buildderef;      var        hp : tdef;        hs : tsym;      begin        { interface definitions }        hp:=tdef(defindex.first);        while assigned(hp) do         begin           hp.buildderef;           hp:=tdef(hp.indexnext);         end;        { interface symbols }        hs:=tsym(symindex.first);        while assigned(hs) do         begin           hs.buildderef;           hs:=tsym(hs.indexnext);         end;      end;    procedure tstoredsymtable.buildderefimpl;      var        hp : tdef;      begin        { definitions }        hp:=tdef(defindex.first);        while assigned(hp) do         begin           hp.buildderefimpl;           hp:=tdef(hp.indexnext);         end;      end;    procedure tstoredsymtable.deref;      var        hp : tdef;        hs : tsym;      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 restype field (PFV) }        hs:=tsym(symindex.first);        while assigned(hs) do         begin           if hs.typ=typesym then             hs.deref;           hs:=tsym(hs.indexnext);         end;        { deref the interface definitions }        hp:=tdef(defindex.first);        while assigned(hp) do         begin           hp.deref;           hp:=tdef(hp.indexnext);         end;        { deref the interface symbols }        hs:=tsym(symindex.first);        while assigned(hs) do         begin           if hs.typ<>typesym then             hs.deref;           hs:=tsym(hs.indexnext);         end;      end;    procedure tstoredsymtable.derefimpl;      var        hp : tdef;        hs: tsym;      begin        { definitions }        hp:=tdef(defindex.first);        while assigned(hp) do         begin           hp.derefimpl;           hp:=tdef(hp.indexnext);         end;        { symbols }        hs:=tsym(symindex.first);        while assigned(hs) do         begin           hs.derefimpl;           hs:=tsym(hs.indexnext);         end;      end;    procedure tstoredsymtable.insert(sym:tsymentry);      var         hsym : tsym;      begin         { set owner and sym indexnb }         sym.owner:=self;         { check the current symtable }         hsym:=tsym(search(sym.name));         if assigned(hsym) then          begin            { in TP and Delphi you can have a local with the              same name as the function, the function is then hidden for              the user. (Under delphi it can still be accessed using result),              but don't allow hiding of RESULT }            if (m_duplicate_names in aktmodeswitches) and               (sym.typ in [localvarsym,paravarsym,absolutevarsym]) and               (vo_is_funcret in tabstractvarsym(sym).varoptions) and               not((m_result in aktmodeswitches) and                   (vo_is_result in tabstractvarsym(sym).varoptions)) then             sym.name:='hidden'+sym.name            else             DuplicateSym(sym,hsym);          end;         { register definition of typesym }         if (sym.typ = typesym) and            assigned(ttypesym(sym).restype.def) then          begin            if not(assigned(ttypesym(sym).restype.def.owner)) and               (ttypesym(sym).restype.def.deftype<>errordef) then              registerdef(ttypesym(sym).restype.def);          end;         { insert in index and search hash }         symindex.insert(sym);         symsearch.insert(sym);      end;    function tstoredsymtable.speedsearch(const s : stringid;speedvalue : cardinal) : tsymentry;      var        hp : Tsym;        newref : tref;      begin        hp:=Tsym(inherited speedsearch(s,speedvalue));        if assigned(hp) then         begin           { reject non static members in static procedures }           if (symtabletype=objectsymtable) and              not(sp_static in hp.symoptions) and              allow_only_static then             Message(sym_e_only_static_in_static);           { unit uses count }           if assigned(current_module) and              (symtabletype=globalsymtable) then             begin               if tglobalsymtable(self).moduleid>=current_module.unitmapsize then                 internalerror(200501152);               inc(current_module.unitmap[tglobalsymtable(self).moduleid].refs);             end;           if make_ref and (cs_browser in aktmoduleswitches) then             begin                newref:=tref.create(hp.lastref,@akttokenpos);                { for symbols that are in tables without browser info or syssyms }                if hp.refcount=0 then                  begin                    hp.defref:=newref;                    hp.lastref:=newref;                  end                else                if resolving_forward and assigned(hp.defref) then                { put it as second reference }                  begin                   newref.nextref:=hp.defref.nextref;                   hp.defref.nextref:=newref;                   hp.lastref.nextref:=nil;                  end                else                  hp.lastref:=newref;                inc(hp.refcount);             end;           if make_ref then               inc(hp.refs);         end; { value was not found }        speedsearch:=hp;      end;{**************************************             Callbacks**************************************}    procedure TStoredSymtable.check_forward(sym : TNamedIndexItem;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).restype.def) and             (ttypesym(sym).restype.def.typesym=ttypesym(sym)) and             (ttypesym(sym).restype.def.deftype=objectdef) then           tobjectdef(ttypesym(sym).restype.def).check_forwards;      end;    procedure TStoredSymtable.labeldefined(p : TNamedIndexItem;arg:pointer);      begin        if (tsym(p).typ=labelsym) and           not(tlabelsym(p).defined) then         begin           if tlabelsym(p).used then            Message1(sym_e_label_used_and_not_defined,tlabelsym(p).realname)           else            Message1(sym_w_label_not_defined,tlabelsym(p).realname);         end;      end;    procedure TStoredSymtable.varsymbolused(p : TNamedIndexItem;arg:pointer);      begin         if (tsym(p).typ in [globalvarsym,localvarsym,paravarsym,fieldvarsym]) and            ((tsym(p).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(p).varoptions = [vo_is_hidden_para]) then             exit;           if (tstoredsym(p).refs=0) then             begin                if (vo_is_funcret in tabstractvarsym(p).varoptions) then                  begin                    { don't warn about the result of constructors }                    if (tsym(p).owner.symtabletype<>localsymtable) or                       (tprocdef(tsym(p).owner.defowner).proctypeoption<>potype_constructor) then                      MessagePos(tsym(p).fileinfo,sym_w_function_result_not_set)                  end                else if (tsym(p).owner.symtabletype=parasymtable) then                  MessagePos1(tsym(p).fileinfo,sym_h_para_identifier_not_used,tsym(p).realname)                else if (tsym(p).owner.symtabletype=objectsymtable) then                  MessagePos2(tsym(p).fileinfo,sym_n_private_identifier_not_used,tsym(p).owner.realname^,tsym(p).realname)                else                  MessagePos1(tsym(p).fileinfo,sym_n_local_identifier_not_used,tsym(p).realname);             end           else if tabstractvarsym(p).varstate in [vs_written,vs_initialised] then             begin                if (tsym(p).owner.symtabletype=parasymtable) then                  begin                    if not(tabstractvarsym(p).varspez in [vs_var,vs_out]) and                       not(vo_is_funcret in tabstractvarsym(p).varoptions) then                      MessagePos1(tsym(p).fileinfo,sym_h_para_identifier_only_set,tsym(p).realname)                  end                else if (tsym(p).owner.symtabletype=objectsymtable) then                  MessagePos2(tsym(p).fileinfo,sym_n_private_identifier_only_set,tsym(p).owner.realname^,tsym(p).realname)                else if not(vo_is_exported in tabstractvarsym(p).varoptions) and                        not(vo_is_funcret in tabstractvarsym(p).varoptions) then                  MessagePos1(tsym(p).fileinfo,sym_n_local_identifier_only_set,tsym(p).realname);             end           else if (tabstractvarsym(p).varstate = vs_read_not_warned) and                   ([vo_is_exported,vo_is_external] * tabstractvarsym(p).varoptions = []) then             MessagePos1(tsym(p).fileinfo,sym_w_identifier_only_read,tsym(p).realname)         end      else if ((tsym(p).owner.symtabletype in              [objectsymtable,parasymtable,localsymtable,staticsymtable])) then          begin           if (Errorcount<>0) or              (sp_internal in tsym(p).symoptions) then             exit;           { do not claim for inherited private fields !! }           if (Tsym(p).refs=0) and (tsym(p).owner.symtabletype=objectsymtable) then             MessagePos2(tsym(p).fileinfo,sym_n_private_method_not_used,tsym(p).owner.realname^,tsym(p).realname)           { units references are problematic }           else            begin              if (Tsym(p).refs=0) and                 not(tsym(p).typ in [enumsym,unitsym]) and                 not(is_funcret_sym(tsym(p))) and                 (                  (tsym(p).typ<>procsym) or                  ((tsym(p).owner.symtabletype=staticsymtable) and                   not current_module.is_unit)                 ) then                MessagePos2(tsym(p).fileinfo,sym_h_local_symbol_not_used,SymTypeName[tsym(p).typ],tsym(p).realname);            end;          end;      end;    procedure TStoredSymtable.TestPrivate(p : TNamedIndexItem;arg:pointer);      begin        if sp_private in tsym(p).symoptions then          varsymbolused(p,arg);      end;    procedure TStoredSymtable.objectprivatesymbolused(p : TNamedIndexItem;arg:pointer);      begin         {           Don't test simple object aliases PM         }         if (tsym(p).typ=typesym) and            (ttypesym(p).restype.def.deftype=objectdef) and            (ttypesym(p).restype.def.typesym=tsym(p)) then           tobjectdef(ttypesym(p).restype.def).symtable.foreach(@TestPrivate,nil);      end;    procedure tstoredsymtable.unchain_overloads(p : TNamedIndexItem;arg:pointer);      begin         if tsym(p).typ=procsym then           tprocsym(p).unchain_overload;      end;    procedure Tstoredsymtable.reset_def(def:Tnamedindexitem;arg:pointer);      begin        Tstoreddef(def).reset;      end;{$ifdef GDB}   function tstoredsymtable.getnewtypecount : word;      begin         getnewtypecount:=pglobaltypecount^;         inc(pglobaltypecount^);      end;{$endif GDB}{***********************************************           Process all entries***********************************************}    procedure Tstoredsymtable.reset_all_defs;      begin        defindex.foreach(@reset_def,nil);      end;    { checks, if all procsyms and methods are defined }    procedure tstoredsymtable.check_forwards;      begin         foreach(@check_forward,nil);      end;    procedure tstoredsymtable.checklabels;      begin         foreach(@labeldefined,nil);      end;    procedure tstoredsymtable.allsymbolsused;      begin         foreach(@varsymbolused,nil);      end;    procedure tstoredsymtable.allprivatesused;      begin         foreach(@objectprivatesymbolused,nil);      end;    procedure tstoredsymtable.unchain_overloaded;      begin         foreach(@unchain_overloads,nil);      end;{$ifdef GDB}    procedure tstoredsymtable.concatstabto(asmlist : taasmoutput);      var        stabstr : Pchar;        p : tsym;      begin        p:=tsym(symindex.first);        while assigned(p) do          begin            { Procsym and typesym are already written }            if not(Tsym(p).typ in [procsym,typesym]) then              begin                if not Tsym(p).isstabwritten then                  begin                    stabstr:=Tsym(p).stabstring;                    if stabstr<>nil then                      asmlist.concat(Tai_stabs.create(stabstr));                    Tsym(p).isstabwritten:=true;                  end;              end;            p:=tsym(p.indexnext);          end;      end;{$endif}    procedure TStoredSymtable._needs_init_final(p : tnamedindexitem;arg:pointer);      begin         if b_needs_init_final then          exit;         case tsym(p).typ of           fieldvarsym,           globalvarsym,           localvarsym,           paravarsym :             begin               if not(is_class(tabstractvarsym(p).vartype.def)) and                  tstoreddef(tabstractvarsym(p).vartype.def).needs_inittable then                 b_needs_init_final:=true;             end;           typedconstsym :             begin               if ttypedconstsym(p).is_writable and                  tstoreddef(ttypedconstsym(p).typedconsttype.def).needs_inittable 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;         foreach(@_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;        recordalignment:=1;        usefieldalignment:=usealign;        padalignment:=1;        { recordalign -1 means C record packing, that starts          with an alignment of 1 }        if usealign=-1 then          fieldalignment:=1        else          fieldalignment:=usealign;      end;    procedure tabstractrecordsymtable.ppuload(ppufile:tcompilerppufile);      var        storesymtable : tsymtable;      begin        storesymtable:=aktrecordsymtable;        aktrecordsymtable:=self;        inherited ppuload(ppufile);        aktrecordsymtable:=storesymtable;      end;    procedure tabstractrecordsymtable.ppuwrite(ppufile:tcompilerppufile);      var        oldtyp : byte;        storesymtable : tsymtable;      begin         storesymtable:=aktrecordsymtable;         aktrecordsymtable:=self;         oldtyp:=ppufile.entrytyp;         ppufile.entrytyp:=subentryid;         inherited ppuwrite(ppufile);         ppufile.entrytyp:=oldtyp;         aktrecordsymtable:=storesymtable;      end;    procedure tabstractrecordsymtable.load_references(ppufile:tcompilerppufile;locals:boolean);      var        storesymtable : tsymtable;      begin        storesymtable:=aktrecordsymtable;        aktrecordsymtable:=self;        inherited load_references(ppufile,locals);        aktrecordsymtable:=storesymtable;      end;    procedure tabstractrecordsymtable.write_references(ppufile:tcompilerppufile;locals:boolean);      var        storesymtable : tsymtable;      begin        storesymtable:=aktrecordsymtable;        aktrecordsymtable:=self;        inherited write_references(ppufile,locals);        aktrecordsymtable:=storesymtable;      end;   procedure tabstractrecordsymtable.derefimpl;     var       storesymtable : tsymtable;     begin       storesymtable:=aktrecordsymtable;       aktrecordsymtable:=self;       inherited derefimpl;       aktrecordsymtable:=storesymtable;     end;    procedure tabstractrecordsymtable.insertfield(sym : tfieldvarsym;addsym:boolean);      var        l      : aint;        varalignrecord,        varalignfield,        varalign : longint;        vardef : tdef;      begin        if addsym then          insert(sym);        { this symbol can't be loaded to a register }        sym.varregable:=vr_none;        { Calculate field offset }        l:=sym.getsize;        vardef:=sym.vartype.def;        varalign:=vardef.alignment;        { Calc the alignment size for C style records }        if (usefieldalignment=-1) then         begin           if (varalign>4) and              ((varalign mod 4)<>0) and              (vardef.deftype=arraydef) then             Message1(sym_w_wrong_C_pack,vardef.typename);           if varalign=0 then             varalign:=l;           if (fieldalignment<aktalignment.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,aktalignment.maxCrecordalign);         end;        if varalign=0 then          varalign:=size_2_align(l);        varalignfield:=used_align(varalign,aktalignment.recordalignmin,fieldalignment);        sym.fieldoffset:=align(datasize,varalignfield);        if (aword(l)+sym.fieldoffset)>high(aint) then          begin            Message(sym_e_segment_too_large);            datasize:=high(aint);          end        else          datasize:=sym.fieldoffset+l;        { Calc alignment needed for this record }        if (usefieldalignment=-1) then          varalignrecord:=used_align(varalign,aktalignment.recordalignmin,aktalignment.maxCrecordalign)        else          if (usefieldalignment=0) then            varalignrecord:=used_align(varalign,aktalignment.recordalignmin,aktalignment.recordalignmax)        else          begin            { packrecords is set explicit, ignore recordalignmax limit }            varalignrecord:=used_align(varalign,aktalignment.recordalignmin,usefieldalignment);          end;        recordalignment:=max(recordalignment,varalignrecord);      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          if usefieldalignment=-1 then            padalignment:=fieldalignment          else            padalignment:=recordalignment;        datasize:=align(datasize,padalignment);      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        ps,nps : tfieldvarsym;        pd,npd : tdef;        varalignrecord,varalign,        storesize,storealign : longint;      begin        storesize:=datasize;        storealign:=fieldalignment;        datasize:=offset;        ps:=tfieldvarsym(unionst.symindex.first);        while assigned(ps) do          begin            nps:=tfieldvarsym(ps.indexnext);            { remove from current symtable }            unionst.symindex.deleteindex(ps);            ps.left:=nil;            ps.right:=nil;            { add to this record }            ps.owner:=self;            datasize:=ps.fieldoffset+offset;            symindex.insert(ps);            symsearch.insert(ps);            { update address }            ps.fieldoffset:=datasize;            { update alignment of this record }            varalign:=ps.vartype.def.alignment;            if varalign=0 then              varalign:=size_2_align(ps.getsize);            varalignrecord:=used_align(varalign,aktalignment.recordalignmin,fieldalignment);            recordalignment:=max(recordalignment,varalignrecord);            { next }            ps:=nps;          end;        pd:=tdef(unionst.defindex.first);        while assigned(pd) do          begin            npd:=tdef(pd.indexnext);            unionst.defindex.deleteindex(pd);            pd.left:=nil;            pd.right:=nil;            registerdef(pd);            pd:=npd;          end;        datasize:=storesize;        fieldalignment:=storealign;      end;{****************************************************************************                              TObjectSymtable****************************************************************************}    constructor tobjectsymtable.create(const n:string;usealign:shortint);      begin        inherited create(n,usealign);        symtabletype:=objectsymtable;      end;    procedure tobjectsymtable.insert(sym:tsymentry);      var         hsym : tsym;      begin         { check for duplicate field id in inherited classes }         if (sym.typ=fieldvarsym) and            assigned(defowner) and            (             not(m_delphi in aktmodeswitches) or             is_object(tdef(defowner))            ) then           begin              { but private ids can be reused }              hsym:=search_class_member(tobjectdef(defowner),sym.name);              if assigned(hsym) and                 tsym(hsym).is_visible_for_object(tobjectdef(defowner),tobjectdef(defowner)) then                DuplicateSym(sym,hsym);           end;         inherited insert(sym);      end;{****************************************************************************                          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;{****************************************************************************                              TLocalSymtable****************************************************************************}    constructor tlocalsymtable.create(level:byte);      begin        inherited create('');        symtabletype:=localsymtable;        symtablelevel:=level;      end;    procedure tlocalsymtable.insert(sym:tsymentry);      var         hsym : tsym;      begin        { need to hide function result? }        hsym:=tsym(search(sym.name));        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 aktmodeswitches) and               (hsym.typ in [absolutevarsym,localvarsym]) and               (vo_is_funcret in tabstractvarsym(hsym).varoptions) and               not((m_result in aktmodeswitches) and                   (vo_is_result in tabstractvarsym(hsym).varoptions)) then              HideSym(hsym)            else              DuplicateSym(sym,hsym);          end;        if assigned(next) and           (next.symtabletype=parasymtable) then          begin            { check para symtable }            hsym:=tsym(next.search(sym.name));            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 aktmodeswitches) and                   (sym.typ in [absolutevarsym,paravarsym]) and                   (vo_is_funcret in tabstractvarsym(sym).varoptions) and                   not((m_result in aktmodeswitches) and                       (vo_is_result in tabstractvarsym(sym).varoptions)) then                  sym.name:='hidden'+sym.name                else                  DuplicateSym(sym,hsym);              end;            { check for duplicate id in local symtable of methods }            if assigned(next.next) and               { funcretsym is allowed !! }               (not is_funcret_sym(sym)) and               (next.next.symtabletype=objectsymtable) then             begin               hsym:=search_class_member(tobjectdef(next.next.defowner),sym.name);               if assigned(hsym) and                 { private ids can be reused }                  (hsym.is_visible_for_object(tobjectdef(next.next.defowner),tobjectdef(next.next.defowner)) or                   (hsym.owner.defowner.owner.symtabletype<>globalsymtable)) then                begin                  { delphi allows to reuse the names in a class, but not                    in object (tp7 compatible) }                  if not((m_delphi in aktmodeswitches) and                         is_class(tdef(next.next.defowner))) then                    DuplicateSym(sym,hsym);                end;             end;          end;         inherited insert(sym);      end;{****************************************************************************                              TParaSymtable****************************************************************************}    constructor tparasymtable.create(level:byte);      begin        inherited create('');        symtabletype:=parasymtable;        symtablelevel:=level;      end;    procedure tparasymtable.insert(sym:tsymentry);      var         hsym : tsym;      begin         { check for duplicate id in para symtable of methods }         if assigned(next) and            (next.symtabletype=objectsymtable) and            { funcretsym is allowed }            (not is_funcret_sym(sym)) then           begin              hsym:=search_class_member(tobjectdef(next.defowner),sym.name);              { private ids can be reused }              if assigned(hsym) and                 Tsym(hsym).is_visible_for_object(tobjectdef(next.defowner),tobjectdef(next.defowner)) then               begin                 { delphi allows to reuse the names in a class, but not                   in object (tp7 compatible) }                 if not((m_delphi in aktmodeswitches) and                        is_class_or_interface(tobjectdef(next.defowner))) then                   DuplicateSym(sym,hsym);               end;           end;         inherited insert(sym);      end;{****************************************************************************                         TAbstractUnitSymtable****************************************************************************}    constructor tabstractunitsymtable.create(const n : string;id:word);      begin        inherited create(n);        moduleid:=id;        symsearch.usehash;{$ifdef GDB}         { reset GDB things }         prev_dbx_counter := dbx_counter;         dbx_counter := nil;         dbx_count := -1;{$endif GDB}      end;    function tabstractunitsymtable.iscurrentunit:boolean;      begin        result:=assigned(current_module) and                (                 (current_module.globalsymtable=self) or                 (current_module.localsymtable=self)                );      end;{$ifdef GDB}      procedure tabstractunitsymtable.concattypestabto(asmlist : taasmoutput);         procedure dowritestabs(asmlist:taasmoutput;st:tsymtable);           var             p : tstoreddef;           begin             p:=tstoreddef(st.defindex.first);             while assigned(p) do               begin                 { also insert local types for the current unit }                 if iscurrentunit then                   begin                     case p.deftype of                       procdef :                         if assigned(tprocdef(p).localst) then                           dowritestabs(asmlist,tprocdef(p).localst);                       objectdef :                         dowritestabs(asmlist,tobjectdef(p).symtable);                     end;                   end;                 if (p.stab_state=stab_state_used) then                   p.concatstabto(asmlist);                 p:=tstoreddef(p.indexnext);               end;           end;        var          old_writing_def_stabs : boolean;          prev_dbx_count : plongint;        begin           if not assigned(name) then             name := stringdup('Main_program');           asmList.concat(tai_comment.Create(strpnew('Begin unit '+name^+' has index '+tostr(moduleid))));           if cs_gdb_dbx in aktglobalswitches then             begin                if dbx_count_ok then                  begin                     asmList.concat(tai_comment.Create(strpnew('"repeated" unit '+name^                              +' has index '+tostr(moduleid)+' dbx count = '+tostr(dbx_count))));                     asmList.concat(Tai_stabs.Create(strpnew('"'+name^+'",'                       +tostr(N_EXCL)+',0,0,'+tostr(dbx_count))));                     exit;                  end                else if not iscurrentunit then                  begin                    prev_dbx_count := dbx_counter;                    dbx_counter := nil;                    do_count_dbx:=false;                    if (symtabletype = globalsymtable) then                      asmList.concat(Tai_stabs.Create(strpnew('"'+name^+'",'+tostr(N_BINCL)+',0,0,0')));                    dbx_counter := @dbx_count;                    dbx_count:=0;                    do_count_dbx:=assigned(dbx_counter);                  end;             end;           old_writing_def_stabs:=writing_def_stabs;           writing_def_stabs:=true;           dowritestabs(asmlist,self);           writing_def_stabs:=old_writing_def_stabs;           if cs_gdb_dbx in aktglobalswitches then             begin                if not iscurrentunit then                  begin                    dbx_counter := prev_dbx_count;                    do_count_dbx:=false;                    asmList.concat(Tai_stabs.Create(strpnew('"'+name^+'",'                      +tostr(N_EINCL)+',0,0,0')));                    do_count_dbx:=assigned(dbx_counter);                    dbx_count_ok := {true}false;                  end;             end;           asmList.concat(tai_comment.Create(strpnew('End unit '+name^+' has index '+tostr(moduleid))));        end;{$endif GDB}{****************************************************************************                              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        next:=symtablestack;        symtablestack:=self;        inherited ppuload(ppufile);        { now we can deref the syms and defs }        deref;        { restore symtablestack }        symtablestack:=next;      end;    procedure tstaticsymtable.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);      end;    procedure tstaticsymtable.load_references(ppufile:tcompilerppufile;locals:boolean);      begin        inherited load_references(ppufile,locals);      end;    procedure tstaticsymtable.write_references(ppufile:tcompilerppufile;locals:boolean);      begin        inherited write_references(ppufile,locals);      end;    procedure tstaticsymtable.insert(sym:tsymentry);      var         hsym : tsym;      begin         { also check the global symtable }         if assigned(next) and            (next.symtabletype=globalsymtable) and            (next.iscurrentunit) then          begin            hsym:=tsym(next.search(sym.name));            if assigned(hsym) then             begin               { Delphi 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_duplicate_names in aktmodeswitches) and                  (hsym.typ=symconst.unitsym) then                HideSym(hsym)               else                DuplicateSym(sym,hsym);             end;          end;         inherited insert(sym);      end;{****************************************************************************                              TGlobalSymtable****************************************************************************}    constructor tglobalsymtable.create(const n : string;id:word);      begin         inherited create(n,id);         symtabletype:=globalsymtable;         symtablelevel:=main_program_level;{$ifdef GDB}         if cs_gdb_dbx in aktglobalswitches then           begin             dbx_count := 0;             unittypecount:=1;             pglobaltypecount := @unittypecount;             {moduleid:=current_module.unitcount;}             {debugList.concat(tai_comment.Create(strpnew('Global '+name^+' has index '+tostr(moduleid))));             debugList.concat(Tai_stabs.Create(strpnew('"'+name^+'",'+tostr(N_BINCL)+',0,0,0')));}             {inc(current_module.unitcount);}             { we can't use dbx_vcount, because we don't know               if the object file will be loaded before or afeter PM }             dbx_count_ok:=false;             dbx_counter:=@dbx_count;             do_count_dbx:=true;           end;{$endif GDB}      end;    procedure tglobalsymtable.ppuload(ppufile:tcompilerppufile);{$ifdef GDB}      var        b : byte;{$endif GDB}      begin{$ifdef GDB}         if cs_gdb_dbx in aktglobalswitches then           begin              UnitTypeCount:=1;              PglobalTypeCount:=@UnitTypeCount;           end;{$endif GDB}         next:=symtablestack;         symtablestack:=self;         inherited ppuload(ppufile);         { now we can deref the syms and defs }         deref;         { restore symtablestack }         symtablestack:=next;         { read dbx count }{$ifdef GDB}        if (current_module.flags and uf_has_dbx)<>0 then         begin           b:=ppufile.readentry;           if b<>ibdbxcount then             Message(unit_f_ppu_dbx_count_problem)           else             dbx_count:=ppufile.getlongint;{$IfDef EXTDEBUG}           writeln('Read dbx_count ',dbx_count,' in unit ',name^,'.ppu');{$ENDIF EXTDEBUG}           { we can't use dbx_vcount, because we don't know             if the object file will be loaded before or afeter PM }           dbx_count_ok := {true}false;         end        else         begin           dbx_count:=-1;           dbx_count_ok:=false;         end;{$endif GDB}      end;    procedure tglobalsymtable.ppuwrite(ppufile:tcompilerppufile);      begin        { write the symtable entries }        inherited ppuwrite(ppufile);        { write dbx count }{$ifdef GDB}        if cs_gdb_dbx in aktglobalswitches then         begin{$IfDef EXTDEBUG}           writeln('Writing dbx_count ',dbx_count,' in unit ',name^,'.ppu');{$ENDIF EXTDEBUG}           ppufile.do_crc:=false;           ppufile.putlongint(dbx_count);           ppufile.writeentry(ibdbxcount);           ppufile.do_crc:=true;         end;{$endif GDB}      end;    procedure tglobalsymtable.load_references(ppufile:tcompilerppufile;locals:boolean);      begin        inherited load_references(ppufile,locals);      end;    procedure tglobalsymtable.write_references(ppufile:tcompilerppufile;locals:boolean);      begin        inherited write_references(ppufile,locals);      end;    procedure tglobalsymtable.insert(sym:tsymentry);      var         hsym : tsym;      begin         hsym:=tsym(search(sym.name));         if assigned(hsym) then          begin            { Delphi 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_duplicate_names in aktmodeswitches) and               (hsym.typ=symconst.unitsym) then             HideSym(hsym)            else             DuplicateSym(sym,hsym);          end;         inherited insert(sym);      end;{$ifdef GDB}   function tglobalsymtable.getnewtypecount : word;      begin         if not (cs_gdb_dbx in aktglobalswitches) then           getnewtypecount:=inherited getnewtypecount         else           begin              getnewtypecount:=unittypecount;              inc(unittypecount);           end;      end;{$endif}{****************************************************************************                              TWITHSYMTABLE****************************************************************************}    constructor twithsymtable.create(aowner:tdef;asymsearch:TDictionary;refnode:pointer{tnode});      begin         inherited create('');         symtabletype:=withsymtable;         withrefnode:=refnode;         { we don't need the symsearch }         symsearch.free;         { set the defaults }         symsearch:=asymsearch;         defowner:=aowner;      end;    destructor twithsymtable.destroy;      begin        tobject(withrefnode).free;        symsearch:=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;{****************************************************************************                          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;    procedure tmacrosymtable.ppuload(ppufile:tcompilerppufile);      begin        next:=macrosymtablestack;        macrosymtablestack:=self;        inherited ppuload(ppufile);        { restore symtablestack }        macrosymtablestack:=next;      end;{*****************************************************************************                             Helper Routines*****************************************************************************}    function findunitsymtable(st:tsymtable):tsymtable;      begin        findunitsymtable:=nil;        repeat          if not assigned(st) then           internalerror(5566561);          case st.symtabletype of            localsymtable,            parasymtable,            staticsymtable :              exit;            globalsymtable :              begin                findunitsymtable:=st;                exit;              end;            objectsymtable :              st:=st.defowner.owner;            recordsymtable :              begin                { don't continue when the current                  symtable is used for variant records }                if trecorddef(st.defowner).isunion then                 begin                   findunitsymtable:=nil;                   exit;                 end                else                 st:=st.defowner.owner;              end;            else              internalerror(5566562);          end;        until false;      end;    function FullTypeName(def,otherdef:tdef):string;      var        s1,s2 : string;      begin        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.deftype=errordef) or           (def1.deftype=errordef) then          exit;        CGMessage2(type_e_incompatible_types,FullTypeName(def1,def2),FullTypeName(def2,def1));      end;    procedure hidesym(sym:tsymentry);      var        s : string;      begin        if assigned(sym.owner) then          sym.owner.rename(sym.name,'hidden'+sym.name)        else          sym.name:='hidden'+sym.name;        s:='hidden'+tsym(sym).realname;        stringdispose(tsym(sym)._realname);        tsym(sym)._realname:=stringdup(s);      end;      var        dupnr : longint; { unique number for duplicate symbols }    procedure duplicatesym(dupsym,sym:tsymentry);      var        st : tsymtable;      begin        Message1(sym_e_duplicate_id,tsym(sym).realname);        st:=findunitsymtable(sym.owner);        with tsym(sym).fileinfo do          begin            if assigned(st) and               (st.symtabletype=globalsymtable) and               (not st.iscurrentunit) then              Message2(sym_h_duplicate_id_where,'unit '+st.name^,tostr(line))            else              Message2(sym_h_duplicate_id_where,current_module.sourcefiles.get_file_name(fileindex),tostr(line));          end;        { Rename duplicate sym to an unreachable name, but it can be          inserted in the symtable without errors }        if assigned(dupsym) then          begin            inc(dupnr);            dupsym.name:='dup'+tostr(dupnr)+dupsym.name;            include(tsym(dupsym).symoptions,sp_implicitrename);          end;      end;{*****************************************************************************                                  Search*****************************************************************************}    function  searchsym(const s : stringid;var srsym:tsym;var srsymtable:tsymtable):boolean;      var        speedvalue : cardinal;        topclass   : tobjectdef;        context : tobjectdef;      begin         speedvalue:=getspeedvalue(s);         srsymtable:=symtablestack;         while assigned(srsymtable) do           begin             srsym:=tsym(srsymtable.speedsearch(s,speedvalue));             if assigned(srsym) then               begin                 topclass:=nil;                 { use the class from withsymtable only when it is                   defined in this unit }                 if (srsymtable.symtabletype=withsymtable) and                    assigned(srsymtable.defowner) and                    (srsymtable.defowner.deftype=objectdef) and                    (srsymtable.defowner.owner.symtabletype in [globalsymtable,staticsymtable]) and                    (srsymtable.defowner.owner.iscurrentunit) then                   topclass:=tobjectdef(srsymtable.defowner)                 else                   begin                     if assigned(current_procinfo) then                       topclass:=current_procinfo.procdef._class;                   end;                 if assigned(current_procinfo) then                   context:=current_procinfo.procdef._class                 else                   context:=nil;                 if tsym(srsym).is_visible_for_object(topclass,context) 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);                     searchsym:=true;                     exit;                   end;               end;             srsymtable:=srsymtable.next;           end;         searchsym:=false;      end;    function  searchsym_type(const s : stringid;var srsym:tsym;var srsymtable:tsymtable):boolean;      var        speedvalue : cardinal;      begin         speedvalue:=getspeedvalue(s);         srsymtable:=symtablestack;         while assigned(srsymtable) do           begin              {                It is not possible to have type defintions in:                  records                  objects                  parameters              }              if not(srsymtable.symtabletype in [recordsymtable,objectsymtable,parasymtable]) then                begin                  srsym:=tsym(srsymtable.speedsearch(s,speedvalue));                  if assigned(srsym) and                     (not assigned(current_procinfo) or                      tsym(srsym).is_visible_for_object(current_procinfo.procdef._class,current_procinfo.procdef._class)) then                    begin                      result:=true;                      exit;                    end                end;              srsymtable:=srsymtable.next;           end;         result:=false;      end;    function  searchsymonlyin(p : tsymtable;const s : stringid):tsym;      var        srsym      : tsym;      begin         { the caller have to take care if srsym=nil }         if assigned(p) then           begin              srsym:=tsym(p.search(s));              if assigned(srsym) then               begin                 searchsymonlyin:=srsym;                 exit;               end;              { also check in the local symtbale if it exists }              if (p.symtabletype=globalsymtable) and                 (p.iscurrentunit) then                begin                   srsym:=tsym(current_module.localsymtable.search(s));                   if assigned(srsym) then                    begin                      searchsymonlyin:=srsym;                      exit;                    end;                end           end;         searchsymonlyin:=nil;       end;    function searchsym_in_class(classh:tobjectdef;const s : stringid):tsym;      var        speedvalue : cardinal;        topclassh  : tobjectdef;        sym        : tsym;      begin         speedvalue:=getspeedvalue(s);         { when the class passed is defined in this unit we           need to use the scope of that class. This is a trick           that can be used to access protected members in other           units. At least kylix supports it this way (PFV) }         if assigned(classh) and            (classh.owner.symtabletype in [globalsymtable,staticsymtable]) and            classh.owner.iscurrentunit then           topclassh:=classh         else           begin             if assigned(current_procinfo) then               topclassh:=current_procinfo.procdef._class             else               topclassh:=nil;           end;         sym:=nil;         while assigned(classh) do          begin            sym:=tsym(classh.symtable.speedsearch(s,speedvalue));            if assigned(sym) and               tsym(sym).is_visible_for_object(topclassh,current_procinfo.procdef._class) then              break            else              sym:=nil;            classh:=classh.childof;          end;         searchsym_in_class:=sym;      end;    function searchsym_in_class_by_msgint(classh:tobjectdef;i:longint):tsym;      var        topclassh  : tobjectdef;        def        : tdef;        sym        : tsym;      begin         { when the class passed is defined in this unit we           need to use the scope of that class. This is a trick           that can be used to access protected members in other           units. At least kylix supports it this way (PFV) }         if assigned(classh) and            (classh.owner.symtabletype in [globalsymtable,staticsymtable]) and            classh.owner.iscurrentunit then           topclassh:=classh         else           begin             if assigned(current_procinfo) then               topclassh:=current_procinfo.procdef._class             else               topclassh:=nil;           end;         sym:=nil;         def:=nil;         while assigned(classh) do          begin            def:=tdef(classh.symtable.defindex.first);            while assigned(def) do             begin               if (def.deftype=procdef) and                  (po_msgint in tprocdef(def).procoptions) and                  (tprocdef(def).messageinf.i=i) then                begin                  sym:=tprocdef(def).procsym;                  if assigned(topclassh) then                   begin                     if tprocdef(def).is_visible_for_object(topclassh) then                      break;                   end                  else                   break;                end;               def:=tdef(def.indexnext);             end;            if assigned(sym) then             break;            classh:=classh.childof;          end;         searchsym_in_class_by_msgint:=sym;      end;    function searchsym_in_class_by_msgstr(classh:tobjectdef;const s:string):tsym;      var        topclassh  : tobjectdef;        def        : tdef;        sym        : tsym;      begin         { when the class passed is defined in this unit we           need to use the scope of that class. This is a trick           that can be used to access protected members in other           units. At least kylix supports it this way (PFV) }         if assigned(classh) and            (classh.owner.symtabletype in [globalsymtable,staticsymtable]) and            classh.owner.iscurrentunit then           topclassh:=classh         else           begin             if assigned(current_procinfo) then               topclassh:=current_procinfo.procdef._class             else               topclassh:=nil;           end;         sym:=nil;         def:=nil;         while assigned(classh) do          begin            def:=tdef(classh.symtable.defindex.first);            while assigned(def) do             begin               if (def.deftype=procdef) and                  (po_msgstr in tprocdef(def).procoptions) and                  (tprocdef(def).messageinf.str=s) then                begin                  sym:=tprocdef(def).procsym;                  if assigned(topclassh) then                   begin                     if tprocdef(def).is_visible_for_object(topclassh) then                      break;                   end                  else                   break;                end;               def:=tdef(def.indexnext);             end;            if assigned(sym) then             break;            classh:=classh.childof;          end;         searchsym_in_class_by_msgstr:=sym;      end;    function search_assignment_operator(from_def,to_def:Tdef):Tprocdef;    var st:Tsymtable;        sym:Tprocsym;        sv:cardinal;        curreq,        besteq : tequaltype;        currpd,        bestpd : tprocdef;    begin      st:=symtablestack;      sv:=getspeedvalue('assign');      besteq:=te_incompatible;      bestpd:=nil;      while st<>nil do        begin          sym:=Tprocsym(st.speedsearch('assign',sv));          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.search_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;          st:=st.next;        end;      result:=bestpd;    end;    function searchsystype(const s: stringid; var srsym: ttypesym): boolean;      var        symowner: tsymtable;      begin        if not(cs_compilesystem in aktmoduleswitches) then          srsym := ttypesym(searchsymonlyin(systemunit,s))        else          searchsym(s,tsym(srsym),symowner);        searchsystype :=          assigned(srsym) and          (srsym.typ = typesym);      end;{$ifdef notused}    function searchsysvar(const s: stringid; var srsym: tsym; var symowner: tsymtable): boolean;      begin        if not(cs_compilesystem in aktmoduleswitches) then          begin            srsym := searchsymonlyin(systemunit,s);            symowner := systemunit;          end        else          searchsym(s,tsym(srsym),symowner);        searchsysvar :=          assigned(srsym) and          (srsym.typ = globalvarsym);      end;{$endif notused}    function search_class_member(pd : tobjectdef;const s : string):tsym;    { searches n in symtable of pd and all anchestors }      var        speedvalue : cardinal;        srsym      : tsym;      begin        speedvalue:=getspeedvalue(s);        while assigned(pd) do         begin           srsym:=tsym(pd.symtable.speedsearch(s,speedvalue));           if assigned(srsym) then            begin              search_class_member:=srsym;              exit;            end;           pd:=pd.childof;         end;        search_class_member:=nil;      end;    function search_macro(const s : string):tsym;      var        p : tsymtable;        speedvalue : cardinal;        srsym      : tsym;      begin        speedvalue:= getspeedvalue(s);        p:=macrosymtablestack;        while assigned(p) do          begin             srsym:=tsym(p.speedsearch(s,speedvalue));             if assigned(srsym) then               begin                 search_macro:= srsym;                 exit;               end;             p:=p.next;          end;        search_macro:= nil;      end;{*****************************************************************************                            Definition Helpers*****************************************************************************}    procedure globaldef(const s : string;var t:ttype);      var st : string;          symt : tsymtable;          srsym      : tsym;          srsymtable : tsymtable;      begin         srsym := nil;         if pos('.',s) > 0 then           begin           st := copy(s,1,pos('.',s)-1);           searchsym(st,srsym,srsymtable);           st := copy(s,pos('.',s)+1,255);           if assigned(srsym) then             begin             if srsym.typ = unitsym then               begin               symt := tunitsym(srsym).unitsymtable;               srsym := tsym(symt.search(st));               end else srsym := nil;             end;           end else st := s;         if srsym = nil then          searchsym(st,srsym,srsymtable);         if srsym = nil then           srsym:=searchsymonlyin(systemunit,st);         if (not assigned(srsym)) or            (srsym.typ<>typesym) then           begin             Message(type_e_type_id_expected);             t:=generrortype;             exit;           end;         t := ttypesym(srsym).restype;      end;{****************************************************************************                              Object Helpers****************************************************************************}    procedure search_class_overloads(aprocsym : tprocsym);    { searches n in symtable of pd and all anchestors }      var        speedvalue : cardinal;        srsym      : tprocsym;        s          : string;        objdef     : tobjectdef;      begin        if aprocsym.overloadchecked then         exit;        aprocsym.overloadchecked:=true;        if (aprocsym.owner.symtabletype<>objectsymtable) then         internalerror(200111021);        objdef:=tobjectdef(aprocsym.owner.defowner);        { we start in the parent }        if not assigned(objdef.childof) then         exit;        objdef:=objdef.childof;        s:=aprocsym.name;        speedvalue:=getspeedvalue(s);        while assigned(objdef) do         begin           srsym:=tprocsym(objdef.symtable.speedsearch(s,speedvalue));           if assigned(srsym) then            begin              if (srsym.typ<>procsym) then               internalerror(200111022);              if srsym.is_visible_for_object(tobjectdef(aprocsym.owner.defowner),tobjectdef(aprocsym.owner.defowner)) then               begin                 srsym.add_para_match_to(Aprocsym,[cpo_ignorehidden,cpo_allowdefaults]);                 { we can stop if the overloads were already added                  for the found symbol }                 if srsym.overloadchecked then                  break;               end;            end;           { next parent }           objdef:=objdef.childof;         end;      end;   procedure tstoredsymtable.testfordefaultproperty(p : TNamedIndexItem;arg:pointer);     begin        if (tsym(p).typ=propertysym) and           (ppo_defaultproperty in tpropertysym(p).propoptions) then          ppointer(arg)^:=p;     end;   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.foreach(@tstoredsymtable(pd.symtable).testfordefaultproperty,@_defaultprop);             if assigned(_defaultprop) then               break;             pd:=pd.childof;          end;        search_default_property:=_defaultprop;     end;{****************************************************************************                              Macro Helpers****************************************************************************}{NOTE: Initially, macrosymtablestack contains initialmacrosymtable.}    procedure def_system_macro(const name : string);      var        mac : tmacro;        s: string;      begin         if name = '' then           internalerror(2004121201);         s:= upper(name);         mac:=tmacro(search_macro(s));         if not assigned(mac) then           begin             mac:=tmacro.create(s);             if macrosymtablestack.symtabletype=localmacrosymtable then               macrosymtablestack.insert(mac)             else               macrosymtablestack.next.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(2004121201);         s:= upper(name);         mac:=tmacro(search_macro(s));         if not assigned(mac) then           begin             mac:=tmacro.create(s);             if macrosymtablestack.symtabletype=localmacrosymtable then               macrosymtablestack.insert(mac)             else               macrosymtablestack.next.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(2004121201);         s:= upper(name);         mac:=tmacro(search_macro(s));         if not assigned(mac) then           begin             mac:=tmacro.create(s);             mac.is_compiler_var:=true;             if macrosymtablestack.symtabletype=localmacrosymtable then               macrosymtablestack.insert(mac)             else               macrosymtablestack.next.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(2004121201);         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^.search(Upper(n)));        if assigned(p) then         getunitalias:=punit_alias(p).newname^        else         getunitalias:=n;      end;{$endif UNITALIASES}{****************************************************************************                            Symtable Stack****************************************************************************}{$ifdef DEBUG}    procedure test_symtablestack;      var         p : tsymtable;         i : longint;      begin         p:=symtablestack;         i:=0;         while assigned(p) do           begin              inc(i);              p:=p.next;              if i>500 then               Message(sym_f_internal_error_in_symtablestack);           end;      end;    procedure list_symtablestack;      var         p : tsymtable;         i : longint;      begin         p:=symtablestack;         i:=0;         while assigned(p) do           begin              inc(i);              writeln(i,' ',p.name^);              p:=p.next;              if i>500 then               Message(sym_f_internal_error_in_symtablestack);           end;      end;{$endif DEBUG}{****************************************************************************                           Init/Done Symtable****************************************************************************}   procedure InitSymtable;     begin       { Reset symbolstack }       registerdef:=false;       symtablestack:=nil;       macrosymtablestack:=nil;       systemunit:=nil;{$ifdef GDB}       globaltypecount:=1;       pglobaltypecount:=@globaltypecount;{$endif GDB}       { create error syms and def }       generrorsym:=terrorsym.create;       generrortype.setdef(terrordef.create);{$ifdef UNITALIASES}       { unit aliases }       unitaliases:=tdictionary.create;{$endif}       initialmacrosymtable:= tmacrosymtable.create(false);       macrosymtablestack:= initialmacrosymtable;       { 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.free;        generrortype.def.free;{$ifdef UNITALIASES}        unitaliases.free;{$endif}        initialmacrosymtable.Free;     end;end.
 |