| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185 | {    Copyright (c) 1998-2002 by Florian Klaempfl and Peter Vreman    This module provides some basic classes    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 cclasses;{$i fpcdefs.inc}{$ifndef VER2_0}  {$define CCLASSESINLINE}{$endif}interface    uses{$IFNDEF USE_FAKE_SYSUTILS}      SysUtils,{$ELSE}      fksysutl,{$ENDIF}      globtype,      CUtils,CStreams;{********************************************                TMemDebug********************************************}    type       tmemdebug = class       private          totalmem,          startmem : int64;          infostr  : string[40];       public          constructor Create(const s:string);          destructor  Destroy;override;          procedure show;          procedure start;          procedure stop;       end;{*******************************************************      TFPList (From rtl/objpas/classes/classesh.inc)********************************************************}const   SListIndexError = 'List index exceeds bounds (%d)';   SListCapacityError = 'The maximum list capacity is reached (%d)';   SListCapacityPower2Error = 'The capacity has to be a power of 2, but is set to %d';   SListCountError = 'List count too large (%d)';type   EListError = class(Exception);const  MaxListSize = Maxint div 16;type  PPointerList = ^TPointerList;  TPointerList = array[0..MaxListSize - 1] of Pointer;  TListSortCompare = function (Item1, Item2: Pointer): Integer;  TListCallback = procedure(data,arg:pointer) of object;  TListStaticCallback = procedure(data,arg:pointer);  TDynStringArray = Array Of String;  TFPList = class(TObject)  private    FList: PPointerList;    FCount: Integer;    FCapacity: Integer;  protected    function Get(Index: Integer): Pointer;    procedure Put(Index: Integer; Item: Pointer);    procedure SetCapacity(NewCapacity: Integer);    procedure SetCount(NewCount: Integer);    Procedure RaiseIndexError(Index : Integer);{$ifndef VER2_6}noreturn;{$endif VER2_6}  public    destructor Destroy; override;    function Add(Item: Pointer): Integer;    procedure Clear;    procedure Delete(Index: Integer);    class procedure Error(const Msg: string; Data: PtrInt);{$ifndef VER2_6}noreturn;{$endif VER2_6}    procedure Exchange(Index1, Index2: Integer);    function Expand: TFPList;    function Extract(item: Pointer): Pointer;    function First: Pointer;    function IndexOf(Item: Pointer): Integer;    procedure Insert(Index: Integer; Item: Pointer);    function Last: Pointer;    procedure Move(CurIndex, NewIndex: Integer);    procedure Assign(Obj:TFPList);    function Remove(Item: Pointer): Integer;    procedure Pack;    procedure Sort(Compare: TListSortCompare);    procedure ForEachCall(proc2call:TListCallback;arg:pointer);    procedure ForEachCall(proc2call:TListStaticCallback;arg:pointer);    property Capacity: Integer read FCapacity write SetCapacity;    property Count: Integer read FCount write SetCount;    property Items[Index: Integer]: Pointer read Get write Put; default;    property List: PPointerList read FList;  end;{*******************************************************        TFPObjectList (From fcl/inc/contnrs.pp)********************************************************}  TObjectListCallback = procedure(data:TObject;arg:pointer) of object;  TObjectListStaticCallback = procedure(data:TObject;arg:pointer);  TFPObjectList = class(TObject)  private    FFreeObjects : Boolean;    FList: TFPList;    function GetCount: integer; {$ifdef CCLASSESINLINE}inline;{$endif}    procedure SetCount(const AValue: integer);  protected    function GetItem(Index: Integer): TObject; {$ifdef CCLASSESINLINE}inline;{$endif}    procedure SetItem(Index: Integer; AObject: TObject);    procedure SetCapacity(NewCapacity: Integer); {$ifdef CCLASSESINLINE}inline;{$endif}    function GetCapacity: integer; {$ifdef CCLASSESINLINE}inline;{$endif}  public    constructor Create;    constructor Create(FreeObjects : Boolean);    destructor Destroy; override;    procedure Clear;    function Add(AObject: TObject): Integer; {$ifdef CCLASSESINLINE}inline;{$endif}    procedure Delete(Index: Integer);    procedure Exchange(Index1, Index2: Integer); {$ifdef CCLASSESINLINE}inline;{$endif}    function Expand: TFPObjectList;{$ifdef CCLASSESINLINE}inline;{$endif}    function Extract(Item: TObject): TObject; {$ifdef CCLASSESINLINE}inline;{$endif}    function Remove(AObject: TObject): Integer;    function IndexOf(AObject: TObject): Integer; {$ifdef CCLASSESINLINE}inline;{$endif}    function FindInstanceOf(AClass: TClass; AExact: Boolean; AStartAt: Integer): Integer;    procedure Insert(Index: Integer; AObject: TObject); {$ifdef CCLASSESINLINE}inline;{$endif}    function First: TObject; {$ifdef CCLASSESINLINE}inline;{$endif}    function Last: TObject; {$ifdef CCLASSESINLINE}inline;{$endif}    procedure Move(CurIndex, NewIndex: Integer); {$ifdef CCLASSESINLINE}inline;{$endif}    procedure Assign(Obj:TFPObjectList);    procedure ConcatListCopy(Obj:TFPObjectList);    procedure Pack; {$ifdef CCLASSESINLINE}inline;{$endif}    procedure Sort(Compare: TListSortCompare); {$ifdef CCLASSESINLINE}inline;{$endif}    procedure ForEachCall(proc2call:TObjectListCallback;arg:pointer); {$ifdef CCLASSESINLINE}inline;{$endif}    procedure ForEachCall(proc2call:TObjectListStaticCallback;arg:pointer); {$ifdef CCLASSESINLINE}inline;{$endif}    property Capacity: Integer read GetCapacity write SetCapacity;    property Count: Integer read GetCount write SetCount;    property OwnsObjects: Boolean read FFreeObjects write FFreeObjects;    property Items[Index: Integer]: TObject read GetItem write SetItem; default;    property List: TFPList read FList;  end;type  THashItem=record    HashValue : LongWord;    StrIndex  : Integer;    NextIndex : Integer;    Data      : Pointer;  end;  PHashItem=^THashItem;const  MaxHashListSize = Maxint div 16;  MaxHashStrSize  = Maxint;  MaxHashTableSize = Maxint div 4;  MaxItemsPerHash = 3;type  PHashItemList = ^THashItemList;  THashItemList = array[0..MaxHashListSize - 1] of THashItem;  PHashTable = ^THashTable;  THashTable = array[0..MaxHashTableSize - 1] of Integer;  TFPHashList = class(TObject)  private    { ItemList }    FHashList     : PHashItemList;    FCount,    FCapacity : Integer;    FCapacityMask: LongWord;    { Hash }    FHashTable    : PHashTable;    FHashCapacity : Integer;    { Strings }{$ifdef symansistr}    FStrs     : PAnsiString;{$else symansistr}    FStrs     : PChar;{$endif symansistr}    FStrCount,    FStrCapacity : Integer;    function InternalFind(AHash:LongWord;const AName:TSymStr;out PrevIndex:Integer):Integer;  protected    function Get(Index: Integer): Pointer;    procedure Put(Index: Integer; Item: Pointer);    procedure SetCapacity(NewCapacity: Integer);    procedure SetCount(NewCount: Integer);    Procedure RaiseIndexError(Index : Integer);    function  AddStr(const s:TSymStr): Integer;    procedure AddToHashTable(Index: Integer);    procedure StrExpand(MinIncSize:Integer);    procedure SetStrCapacity(NewCapacity: Integer);    procedure SetHashCapacity(NewCapacity: Integer);    procedure ReHash;  public    constructor Create;    destructor Destroy; override;    function Add(const AName:TSymStr;Item: Pointer): Integer;    procedure Clear;    function NameOfIndex(Index: Integer): TSymStr;    function HashOfIndex(Index: Integer): LongWord;    function GetNextCollision(Index: Integer): Integer;    procedure Delete(Index: Integer);    class procedure Error(const Msg: string; Data: PtrInt);{$ifndef VER2_6}noreturn;{$endif VER2_6}    function Expand: TFPHashList;    function Extract(item: Pointer): Pointer;    function IndexOf(Item: Pointer): Integer;    function Find(const AName:TSymStr): Pointer;    function FindIndexOf(const AName:TSymStr): Integer;    function FindWithHash(const AName:TSymStr;AHash:LongWord): Pointer;    function Rename(const AOldName,ANewName:TSymStr): Integer;    function Remove(Item: Pointer): Integer;    procedure Pack;    procedure ShowStatistics;    procedure ForEachCall(proc2call:TListCallback;arg:pointer);    procedure ForEachCall(proc2call:TListStaticCallback;arg:pointer);    procedure WhileEachCall(proc2call:TListCallback;arg:pointer);    procedure WhileEachCall(proc2call:TListStaticCallback;arg:pointer);    property Capacity: Integer read FCapacity write SetCapacity;    property Count: Integer read FCount write SetCount;    property Items[Index: Integer]: Pointer read Get write Put; default;    property List: PHashItemList read FHashList;{$ifdef symansistr}    property Strs: PSymStr read FStrs;{$else}    property Strs: PChar read FStrs;{$endif}  end;{*******************************************************        TFPHashObjectList (From fcl/inc/contnrs.pp)********************************************************}  TFPHashObjectList = class;  { TFPHashObject }  TFPHashObject = class  private    FOwner     : TFPHashObjectList;    FStrIndex  : Integer;    procedure InternalChangeOwner(HashObjectList:TFPHashObjectList;const s:TSymStr);  protected    function GetName:TSymStr;virtual;    function GetHash:Longword;virtual;  public    constructor CreateNotOwned;    constructor Create(HashObjectList:TFPHashObjectList;const s:TSymStr);    procedure ChangeOwner(HashObjectList:TFPHashObjectList);    procedure ChangeOwnerAndName(HashObjectList:TFPHashObjectList;const s:TSymStr); {$ifdef CCLASSESINLINE}inline;{$endif}    procedure Rename(const ANewName:TSymStr);    property Name:TSymStr read GetName;    property Hash:Longword read GetHash;    property OwnerList: TFPHashObjectList read FOwner;  end;  TFPHashObjectList = class(TObject)  private    FFreeObjects : Boolean;    FHashList: TFPHashList;    function GetCount: integer; {$ifdef CCLASSESINLINE}inline;{$endif}    procedure SetCount(const AValue: integer);  protected    function GetItem(Index: Integer): TObject; {$ifdef CCLASSESINLINE}inline;{$endif}    procedure SetItem(Index: Integer; AObject: TObject);    procedure SetCapacity(NewCapacity: Integer); {$ifdef CCLASSESINLINE}inline;{$endif}    function GetCapacity: integer; {$ifdef CCLASSESINLINE}inline;{$endif}  public    constructor Create(FreeObjects : boolean = True);    destructor Destroy; override;    procedure Clear;    function Add(const AName:TSymStr;AObject: TObject): Integer; {$ifdef CCLASSESINLINE}inline;{$endif}    function NameOfIndex(Index: Integer): TSymStr; {$ifdef CCLASSESINLINE}inline;{$endif}    function HashOfIndex(Index: Integer): LongWord; {$ifdef CCLASSESINLINE}inline;{$endif}    function GetNextCollision(Index: Integer): Integer; {$ifdef CCLASSESINLINE}inline;{$endif}    procedure Delete(Index: Integer);    function Expand: TFPHashObjectList; {$ifdef CCLASSESINLINE}inline;{$endif}    function Extract(Item: TObject): TObject; {$ifdef CCLASSESINLINE}inline;{$endif}    function Remove(AObject: TObject): Integer;    function IndexOf(AObject: TObject): Integer; {$ifdef CCLASSESINLINE}inline;{$endif}    function Find(const s:TSymStr): TObject; {$ifdef CCLASSESINLINE}inline;{$endif}    function FindIndexOf(const s:TSymStr): Integer; {$ifdef CCLASSESINLINE}inline;{$endif}    function FindWithHash(const AName:TSymStr;AHash:LongWord): Pointer;    function Rename(const AOldName,ANewName:TSymStr): Integer; {$ifdef CCLASSESINLINE}inline;{$endif}    function FindInstanceOf(AClass: TClass; AExact: Boolean; AStartAt: Integer): Integer;    procedure Pack; {$ifdef CCLASSESINLINE}inline;{$endif}    procedure ShowStatistics; {$ifdef CCLASSESINLINE}inline;{$endif}    procedure ForEachCall(proc2call:TObjectListCallback;arg:pointer); {$ifdef CCLASSESINLINE}inline;{$endif}    procedure ForEachCall(proc2call:TObjectListStaticCallback;arg:pointer); {$ifdef CCLASSESINLINE}inline;{$endif}    procedure WhileEachCall(proc2call:TObjectListCallback;arg:pointer); {$ifdef CCLASSESINLINE}inline;{$endif}    procedure WhileEachCall(proc2call:TObjectListStaticCallback;arg:pointer); {$ifdef CCLASSESINLINE}inline;{$endif}    property Capacity: Integer read GetCapacity write SetCapacity;    property Count: Integer read GetCount write SetCount;    property OwnsObjects: Boolean read FFreeObjects write FFreeObjects;    property Items[Index: Integer]: TObject read GetItem write SetItem; default;    property List: TFPHashList read FHashList;  end;{********************************************                TLinkedList********************************************}    type       TLinkedListItem = class       public          Previous,          Next : TLinkedListItem;          Constructor Create;          Destructor Destroy;override;          Function GetCopy:TLinkedListItem;virtual;       end;       TLinkedListItemClass = class of TLinkedListItem;       TLinkedList = class       private          FCount : integer;          FFirst,          FLast  : TLinkedListItem;          FNoClear : boolean;       public          constructor Create;          destructor  Destroy;override;          { true when the List is empty }          function  Empty:boolean; {$ifdef CCLASSESINLINE}inline;{$endif}          { deletes all Items }          procedure Clear;          { inserts an Item }          procedure Insert(Item:TLinkedListItem);          { inserts an Item before Loc }          procedure InsertBefore(Item,Loc : TLinkedListItem);          { inserts an Item after Loc }          procedure InsertAfter(Item,Loc : TLinkedListItem);virtual;          { concats an Item }          procedure Concat(Item:TLinkedListItem);          { deletes an Item }          procedure Remove(Item:TLinkedListItem);          { Gets First Item }          function  GetFirst:TLinkedListItem;          { Gets last Item }          function  GetLast:TLinkedListItem;          { inserts another List at the begin and make this List empty }          procedure insertList(p : TLinkedList);          { inserts another List before the provided item and make this List empty }          procedure insertListBefore(Item:TLinkedListItem;p : TLinkedList);          { inserts another List after the provided item and make this List empty }          procedure insertListAfter(Item:TLinkedListItem;p : TLinkedList);          { concats another List at the end and make this List empty }          procedure concatList(p : TLinkedList);          { concats another List at the start and makes a copy            the list is ordered in reverse.          }          procedure insertListcopy(p : TLinkedList);          { concats another List at the end and makes a copy }          procedure concatListcopy(p : TLinkedList);          { removes all items from the list, the items are not freed }          procedure RemoveAll;          property First:TLinkedListItem read FFirst;          property Last:TLinkedListItem read FLast;          property Count:Integer read FCount;          property NoClear:boolean write FNoClear;       end;{********************************************                TCmdStrList********************************************}       { string containerItem }       TCmdStrListItem = class(TLinkedListItem)          FPStr : TCmdStr;       public          constructor Create(const s:TCmdStr);          destructor  Destroy;override;          function GetCopy:TLinkedListItem;override;          function Str:TCmdStr; {$ifdef CCLASSESINLINE}inline;{$endif}       end;       { string container }       TCmdStrList = class(TLinkedList)       private          FDoubles : boolean;  { if this is set to true, doubles (case insensitive!) are allowed }       public          constructor Create;          constructor Create_No_Double;          { inserts an Item }          procedure Insert(const s:TCmdStr);          { concats an Item }          procedure Concat(const s:TCmdStr);          { deletes an Item }          procedure Remove(const s:TCmdStr);          { Gets First Item }          function  GetFirst:TCmdStr;          { Gets last Item }          function  GetLast:TCmdStr;          { true if string is in the container, compare case sensitive }          function FindCase(const s:TCmdStr):TCmdStrListItem;          { true if string is in the container }          function Find(const s:TCmdStr):TCmdStrListItem;          { inserts an item }          procedure InsertItem(item:TCmdStrListItem); {$ifdef CCLASSESINLINE}inline;{$endif}          { concats an item }          procedure ConcatItem(item:TCmdStrListItem); {$ifdef CCLASSESINLINE}inline;{$endif}          property Doubles:boolean read FDoubles write FDoubles;       end;{********************************************              DynamicArray********************************************}     type       { can't use sizeof(integer) because it crashes gdb }       tdynamicblockdata=array[0..1024*1024-1] of byte;       pdynamicblock = ^tdynamicblock;       tdynamicblock = record         pos,         size,         used : longword;         Next : pdynamicblock;         data : tdynamicblockdata;       end;     const       dynamicblockbasesize = sizeof(tdynamicblock)-sizeof(tdynamicblockdata);       mindynamicblocksize = 8*sizeof(pointer);     type       tdynamicarray = class       private         FPosn       : longword;         FPosnblock  : pdynamicblock;         FCurrBlocksize,         FMaxBlocksize  : longword;         FFirstblock,         FLastblock  : pdynamicblock;         procedure grow;       public         constructor Create(Ablocksize:longword);         destructor  Destroy;override;         procedure reset;         function  size:longword;         procedure align(i:longword);         procedure seek(i:longword);         function  read(var d;len:longword):longword;         procedure write(const d;len:longword);         procedure writestr(const s:string); {$ifdef CCLASSESINLINE}inline;{$endif}         procedure readstream(f:TCStream;maxlen:longword);         procedure writestream(f:TCStream);         function  equal(other:tdynamicarray):boolean;         property  CurrBlockSize : longword read FCurrBlocksize;         property  FirstBlock : PDynamicBlock read FFirstBlock;         property  Pos : longword read FPosn;       end;{******************************************************************   THashSet (keys not limited to ShortString, no indexed access)*******************************************************************}       PPHashSetItem = ^PHashSetItem;       PHashSetItem = ^THashSetItem;       THashSetItem = record         Next: PHashSetItem;         Key: Pointer;         KeyLength: Integer;         HashValue: LongWord;         Data: TObject;       end;       THashSet = class(TObject)       private         FCount: LongWord;         FOwnsObjects: Boolean;         FOwnsKeys: Boolean;         function Lookup(Key: Pointer; KeyLen: Integer; var Found: Boolean;           CanCreate: Boolean): PHashSetItem;         procedure Resize(NewCapacity: LongWord);       protected         FBucket: PPHashSetItem;         FBucketCount: LongWord;         class procedure FreeItem(item:PHashSetItem); virtual;         class function SizeOfItem: Integer; virtual;       public         constructor Create(InitSize: Integer; OwnKeys, OwnObjects: Boolean);         destructor Destroy; override;         procedure Clear;         { finds an entry by key }         function Find(Key: Pointer; KeyLen: Integer): PHashSetItem;virtual;         { finds an entry, creates one if not exists }         function FindOrAdd(Key: Pointer; KeyLen: Integer;           var Found: Boolean): PHashSetItem;virtual;         { finds an entry, creates one if not exists }         function FindOrAdd(Key: Pointer; KeyLen: Integer): PHashSetItem;virtual;         { returns Data by given Key }         function Get(Key: Pointer; KeyLen: Integer): TObject;virtual;         { removes an entry, returns False if entry wasn't there }         function Remove(Entry: PHashSetItem): Boolean;         property Count: LongWord read FCount;       end;{******************************************************************                             TTagHasSet*******************************************************************}       PPTagHashSetItem = ^PTagHashSetItem;       PTagHashSetItem = ^TTagHashSetItem;       TTagHashSetItem = record         Next: PTagHashSetItem;         Key: Pointer;         KeyLength: Integer;         HashValue: LongWord;         Data: TObject;         Tag: LongWord;       end;       TTagHashSet = class(THashSet)       private         function Lookup(Key: Pointer; KeyLen: Integer; Tag: LongWord; var Found: Boolean;           CanCreate: Boolean): PTagHashSetItem;       protected         class procedure FreeItem(item:PHashSetItem); override;         class function SizeOfItem: Integer; override;       public         { finds an entry by key }         function Find(Key: Pointer; KeyLen: Integer; Tag: LongWord): PTagHashSetItem; reintroduce;         { finds an entry, creates one if not exists }         function FindOrAdd(Key: Pointer; KeyLen: Integer; Tag: LongWord;           var Found: Boolean): PTagHashSetItem; reintroduce;         { finds an entry, creates one if not exists }         function FindOrAdd(Key: Pointer; KeyLen: Integer; Tag: LongWord): PTagHashSetItem; reintroduce;         { returns Data by given Key }         function Get(Key: Pointer; KeyLen: Integer; Tag: LongWord): TObject; reintroduce;       end;{******************************************************************                             tbitset*******************************************************************}       tbitset = class       private         fdata: pbyte;         fdatasize: longint;       public         constructor create(initsize: longint);         constructor create_bytesize(bytesize: longint);         destructor destroy; override;         procedure clear;         procedure grow(nsize: longint);         { sets a bit }         procedure include(index: longint);         { clears a bit }         procedure exclude(index: longint);         { finds an entry, creates one if not exists }         function isset(index: longint): boolean;         procedure addset(aset: tbitset);         procedure subset(aset: tbitset);         property data: pbyte read fdata;         property datasize: longint read fdatasize;      end;    function FPHash(P: PChar; Len: Integer; Tag: LongWord): LongWord;    function FPHash(P: PChar; Len: Integer): LongWord; inline;    function FPHash(const s:shortstring):LongWord; inline;    function FPHash(const a:ansistring):LongWord; inline;    function ExtractStrings(Separators, WhiteSpace: TSysCharSet; Content: PChar; var Strings: TDynStringArray; AddEmptyStrings : Boolean = False): Integer;implementation{*****************************************************************************                                    Memory debug*****************************************************************************}    function ExtractStrings(Separators, WhiteSpace: TSysCharSet; Content: PChar; var Strings: TDynStringArray; AddEmptyStrings : Boolean = False): Integer;    var      b, c : pchar;      procedure SkipWhitespace;        begin          while (c^ in Whitespace) do            inc (c);        end;      procedure AddString;        var          l : integer;          s : string;        begin          l := c-b;          s:='';          if (l > 0) or AddEmptyStrings then            begin              setlength(s, l);              if l>0 then                move (b^, s[1],l*SizeOf(char));              l:=length(Strings);              setlength(Strings,l+1);              Strings[l]:=S;              inc (result);            end;        end;    var      quoted : char;    begin      result := 0;      c := Content;      Quoted := #0;      Separators := Separators + [#13, #10] - ['''','"'];      SkipWhitespace;      b := c;      while (c^ <> #0) do        begin          if (c^ = Quoted) then            begin              if ((c+1)^ = Quoted) then                inc (c)              else                Quoted := #0            end          else if (Quoted = #0) and (c^ in ['''','"']) then            Quoted := c^;          if (Quoted = #0) and (c^ in Separators) then            begin              AddString;              inc (c);              SkipWhitespace;              b := c;            end          else            inc (c);        end;      if (c <> b) then        AddString;    end;    constructor tmemdebug.create(const s:string);      begin        infostr:=s;        totalmem:=0;        Start;      end;    procedure tmemdebug.start;      var        status : TFPCHeapStatus;      begin        status:=GetFPCHeapStatus;        startmem:=status.CurrHeapUsed;      end;    procedure tmemdebug.stop;      var        status : TFPCHeapStatus;      begin        if startmem<>0 then         begin           status:=GetFPCHeapStatus;           inc(TotalMem,startmem-status.CurrHeapUsed);           startmem:=0;         end;      end;    destructor tmemdebug.destroy;      begin        Stop;        show;      end;    procedure tmemdebug.show;      begin        write('memory [',infostr,'] ');        if TotalMem>0 then         writeln(DStr(TotalMem shr 10),' Kb released')        else         writeln(DStr((-TotalMem) shr 10),' Kb allocated');      end;{*****************************************************************************               TFPObjectList (Copied from rtl/objpas/classes/lists.inc)*****************************************************************************}procedure TFPList.RaiseIndexError(Index : Integer);{$ifndef VER2_6}noreturn;{$endif VER2_6}begin  Error(SListIndexError, Index);end;function TFPList.Get(Index: Integer): Pointer;begin  If (Index < 0) or (Index >= FCount) then    RaiseIndexError(Index);  Result:=FList^[Index];end;procedure TFPList.Put(Index: Integer; Item: Pointer);begin  if (Index < 0) or (Index >= FCount) then    RaiseIndexError(Index);  Flist^[Index] := Item;end;function TFPList.Extract(item: Pointer): Pointer;var  i : Integer;begin  result := nil;  i := IndexOf(item);  if i >= 0 then   begin     Result := item;     FList^[i] := nil;     Delete(i);   end;end;procedure TFPList.SetCapacity(NewCapacity: Integer);begin  If (NewCapacity < FCount) or (NewCapacity > MaxListSize) then     Error (SListCapacityError, NewCapacity);  if NewCapacity = FCapacity then    exit;  ReallocMem(FList, SizeOf(Pointer)*NewCapacity);  FCapacity := NewCapacity;end;procedure TFPList.SetCount(NewCount: Integer);begin  if (NewCount < 0) or (NewCount > MaxListSize)then    Error(SListCountError, NewCount);  If NewCount > FCount then    begin    If NewCount > FCapacity then      SetCapacity(NewCount);    If FCount < NewCount then      FillChar(Flist^[FCount], (NewCount-FCount) *  sizeof(Pointer), 0);    end;  FCount := Newcount;end;destructor TFPList.Destroy;begin  Self.Clear;  inherited Destroy;end;function TFPList.Add(Item: Pointer): Integer;begin  if FCount = FCapacity then    Self.Expand;  FList^[FCount] := Item;  Result := FCount;  inc(FCount);end;procedure TFPList.Clear;begin  if Assigned(FList) then  begin    SetCount(0);    SetCapacity(0);    FList := nil;  end;end;procedure TFPList.Delete(Index: Integer);begin  If (Index<0) or (Index>=FCount) then    Error (SListIndexError, Index);  dec(FCount);  System.Move (FList^[Index+1], FList^[Index], (FCount - Index) * SizeOf(Pointer));  { Shrink the list if appropriate }  if (FCapacity > 256) and (FCount < FCapacity shr 2) then  begin    FCapacity := FCapacity shr 1;    ReallocMem(FList, SizeOf(Pointer) * FCapacity);  end;end;class procedure TFPList.Error(const Msg: string; Data: PtrInt);{$ifndef VER2_6}noreturn;{$endif VER2_6}begin  Raise EListError.CreateFmt(Msg,[Data]) at get_caller_addr(get_frame), get_caller_frame(get_frame);end;procedure TFPList.Exchange(Index1, Index2: Integer);var  Temp : Pointer;begin  If ((Index1 >= FCount) or (Index1 < 0)) then    Error(SListIndexError, Index1);  If ((Index2 >= FCount) or (Index2 < 0)) then    Error(SListIndexError, Index2);  Temp := FList^[Index1];  FList^[Index1] := FList^[Index2];  FList^[Index2] := Temp;end;function TFPList.Expand: TFPList;var  IncSize : Longint;begin  Result := Self;  if FCount < FCapacity then    exit;  IncSize := sizeof(ptrint)*2;  if FCapacity > 127 then    Inc(IncSize, FCapacity shr 2)  else if FCapacity > sizeof(ptrint)*4 then    Inc(IncSize, FCapacity shr 1)  else if FCapacity >= sizeof(ptrint) then    inc(IncSize,sizeof(ptrint));  SetCapacity(FCapacity + IncSize);end;function TFPList.First: Pointer;begin  If FCount<>0 then    Result := Items[0]  else    Result := Nil;end;function TFPList.IndexOf(Item: Pointer): Integer;var  psrc  : PPointer;  Index : Integer;begin  Result:=-1;  psrc:=@FList^[0];  For Index:=0 To FCount-1 Do    begin      if psrc^=Item then        begin          Result:=Index;          exit;        end;      inc(psrc);    end;end;procedure TFPList.Insert(Index: Integer; Item: Pointer);begin  if (Index < 0) or (Index > FCount )then    Error(SlistIndexError, Index);  iF FCount = FCapacity then Self.Expand;  if Index<FCount then    System.Move(Flist^[Index], Flist^[Index+1], (FCount - Index) * SizeOf(Pointer));  FList^[Index] := Item;  FCount := FCount + 1;end;function TFPList.Last: Pointer;begin  If FCount<>0 then    Result := Items[FCount - 1]  else    Result := nilend;procedure TFPList.Move(CurIndex, NewIndex: Integer);var  Temp : Pointer;begin  if ((CurIndex < 0) or (CurIndex > Count - 1)) then    Error(SListIndexError, CurIndex);  if (NewINdex < 0) then    Error(SlistIndexError, NewIndex);  Temp := FList^[CurIndex];  FList^[CurIndex] := nil;  Self.Delete(CurIndex);  Self.Insert(NewIndex, nil);  FList^[NewIndex] := Temp;end;function TFPList.Remove(Item: Pointer): Integer;begin  Result := IndexOf(Item);  If Result <> -1 then    Self.Delete(Result);end;procedure TFPList.Pack;var  NewCount,  i : integer;  pdest,  psrc : PPointer;begin  NewCount:=0;  psrc:=@FList^[0];  pdest:=psrc;  For I:=0 To FCount-1 Do    begin      if assigned(psrc^) then        begin          pdest^:=psrc^;          inc(pdest);          inc(NewCount);        end;      inc(psrc);    end;  FCount:=NewCount;end;Procedure QuickSort(FList: PPointerList; L, R : Longint;Compare: TListSortCompare);var  I, J, P: Longint;  PItem, Q : Pointer;begin repeat   I := L;   J := R;   P := (L + R) div 2;   repeat     PItem := FList^[P];     while Compare(PItem, FList^[i]) > 0 do       I := I + 1;     while Compare(PItem, FList^[J]) < 0 do       J := J - 1;     If I <= J then     begin       Q := FList^[I];       Flist^[I] := FList^[J];       FList^[J] := Q;       if P = I then        P := J       else if P = J then        P := I;       I := I + 1;       J := J - 1;     end;   until I > J;   if L < J then     QuickSort(FList, L, J, Compare);   L := I; until I >= R;end;procedure TFPList.Sort(Compare: TListSortCompare);begin  if Not Assigned(FList) or (FCount < 2) then exit;  QuickSort(Flist, 0, FCount-1, Compare);end;procedure TFPList.Assign(Obj: TFPList);var  i: Integer;begin  Clear;  for I := 0 to Obj.Count - 1 do    Add(Obj[i]);end;procedure TFPList.ForEachCall(proc2call:TListCallback;arg:pointer);var  i : integer;  p : pointer;begin  For I:=0 To Count-1 Do    begin      p:=FList^[i];      if assigned(p) then        proc2call(p,arg);    end;end;procedure TFPList.ForEachCall(proc2call:TListStaticCallback;arg:pointer);var  i : integer;  p : pointer;begin  For I:=0 To Count-1 Do    begin      p:=FList^[i];      if assigned(p) then        proc2call(p,arg);    end;end;{*****************************************************************************            TFPObjectList (Copied from rtl/objpas/classes/lists.inc)*****************************************************************************}constructor TFPObjectList.Create(FreeObjects : boolean);begin  Create;  FFreeObjects := Freeobjects;end;destructor TFPObjectList.Destroy;begin  if (FList <> nil) then  begin    Clear;    FList.Destroy;    FList:=nil;  end;  inherited Destroy;end;procedure TFPObjectList.Clear;var  i: integer;begin  if FFreeObjects then    for i := 0 to FList.Count - 1 do      TObject(FList[i]).Free;  FList.Clear;end;constructor TFPObjectList.Create;begin  inherited Create;  FList := TFPList.Create;  FFreeObjects := True;end;function TFPObjectList.GetCount: integer;begin  Result := FList.Count;end;procedure TFPObjectList.SetCount(const AValue: integer);begin  if FList.Count <> AValue then    FList.Count := AValue;end;function TFPObjectList.GetItem(Index: Integer): TObject;begin  Result := TObject(FList[Index]);end;procedure TFPObjectList.SetItem(Index: Integer; AObject: TObject);begin  if OwnsObjects then    TObject(FList[Index]).Free;  FList[index] := AObject;end;procedure TFPObjectList.SetCapacity(NewCapacity: Integer);begin  FList.Capacity := NewCapacity;end;function TFPObjectList.GetCapacity: integer;begin  Result := FList.Capacity;end;function TFPObjectList.Add(AObject: TObject): Integer;begin  Result := FList.Add(AObject);end;procedure TFPObjectList.Delete(Index: Integer);begin  if OwnsObjects then    TObject(FList[Index]).Free;  FList.Delete(Index);end;procedure TFPObjectList.Exchange(Index1, Index2: Integer);begin  FList.Exchange(Index1, Index2);end;function TFPObjectList.Expand: TFPObjectList;begin  FList.Expand;  Result := Self;end;function TFPObjectList.Extract(Item: TObject): TObject;begin  Result := TObject(FList.Extract(Item));end;function TFPObjectList.Remove(AObject: TObject): Integer;begin  Result := IndexOf(AObject);  if (Result <> -1) then  begin    if OwnsObjects then      TObject(FList[Result]).Free;    FList.Delete(Result);  end;end;function TFPObjectList.IndexOf(AObject: TObject): Integer;begin  Result := FList.IndexOf(Pointer(AObject));end;function TFPObjectList.FindInstanceOf(AClass: TClass; AExact: Boolean; AStartAt : Integer): Integer;var  I : Integer;begin  I:=AStartAt;  Result:=-1;  If AExact then    while (I<Count) and (Result=-1) do      If Items[i].ClassType=AClass then        Result:=I      else        Inc(I)  else    while (I<Count) and (Result=-1) do      If Items[i].InheritsFrom(AClass) then        Result:=I      else        Inc(I);end;procedure TFPObjectList.Insert(Index: Integer; AObject: TObject);begin  FList.Insert(Index, Pointer(AObject));end;procedure TFPObjectList.Move(CurIndex, NewIndex: Integer);begin  FList.Move(CurIndex, NewIndex);end;procedure TFPObjectList.Assign(Obj: TFPObjectList);begin  Clear;  ConcatListCopy(Obj);end;procedure TFPObjectList.ConcatListCopy(Obj: TFPObjectList);var  i: Integer;begin  for I := 0 to Obj.Count - 1 do    Add(Obj[i]);end;procedure TFPObjectList.Pack;begin  FList.Pack;end;procedure TFPObjectList.Sort(Compare: TListSortCompare);begin  FList.Sort(Compare);end;function TFPObjectList.First: TObject;begin  Result := TObject(FList.First);end;function TFPObjectList.Last: TObject;begin  Result := TObject(FList.Last);end;procedure TFPObjectList.ForEachCall(proc2call:TObjectListCallback;arg:pointer);begin  FList.ForEachCall(TListCallBack(proc2call),arg);end;procedure TFPObjectList.ForEachCall(proc2call:TObjectListStaticCallback;arg:pointer);begin  FList.ForEachCall(TListStaticCallBack(proc2call),arg);end;{*****************************************************************************                            TFPHashList*****************************************************************************}    function FPHash(P: PChar; Len: Integer; Tag: LongWord): LongWord;    Var      pmax : pchar;    begin{$push}{$q-,r-}      result:=Tag;      pmax:=p+len;      while (p<pmax) do        begin          {DJBHash: result:=result*33 + next_char}          result:=LongWord(LongInt(result shl 5) + LongInt(result)) + LongWord(P^);          inc(p);        end;{$pop}    end;    function FPHash(P: PChar; Len: Integer): LongWord; inline;    begin      result:=fphash(P,Len, 5381);    end;    function FPHash(const s: shortstring): LongWord; inline;    begin      result:=fphash(pchar(@s[1]),length(s));    end;    function FPHash(const a: ansistring): LongWord; inline;    begin      result:=fphash(pchar(a),length(a));    end;procedure TFPHashList.RaiseIndexError(Index : Integer);begin  Error(SListIndexError, Index);end;function TFPHashList.Get(Index: Integer): Pointer;begin  If (Index < 0) or (Index >= FCount) then    RaiseIndexError(Index);  Result:=FHashList^[Index].Data;end;procedure TFPHashList.Put(Index: Integer; Item: Pointer);begin  if (Index < 0) or (Index >= FCount) then    RaiseIndexError(Index);  FHashList^[Index].Data:=Item;end;function TFPHashList.NameOfIndex(Index: Integer): TSymStr;begin  If (Index < 0) or (Index >= FCount) then    RaiseIndexError(Index);  with FHashList^[Index] do    begin      if StrIndex>=0 then        Result:=PSymStr(@FStrs[StrIndex])^      else        Result:='';    end;end;function TFPHashList.HashOfIndex(Index: Integer): LongWord;begin  If (Index < 0) or (Index >= FCount) then    RaiseIndexError(Index);  Result:=FHashList^[Index].HashValue;end;function TFPHashList.GetNextCollision(Index: Integer): Integer;begin  Result:=-1;  if ((Index > -1) and (Index < FCount)) then    Result:=FHashList^[Index].NextIndex;end;function TFPHashList.Extract(item: Pointer): Pointer;var  i : Integer;begin  result := nil;  i := IndexOf(item);  if i >= 0 then   begin     Result := item;     Delete(i);   end;end;procedure TFPHashList.SetCapacity(NewCapacity: Integer);var  power: longint;begin  { use a power of two to be able to quickly calculate the hash table index }  if NewCapacity <> 0 then    NewCapacity := nextpowerof2((NewCapacity+(MaxItemsPerHash-1)) div MaxItemsPerHash, power) * MaxItemsPerHash;  if (NewCapacity < FCount) or (NewCapacity > MaxHashListSize) then     Error (SListCapacityError, NewCapacity);  if NewCapacity = FCapacity then    exit;  ReallocMem(FHashList, NewCapacity*SizeOf(THashItem));  FCapacity := NewCapacity;  { Maybe expand hash also }  if FCapacity>FHashCapacity*MaxItemsPerHash then    SetHashCapacity(FCapacity div MaxItemsPerHash);end;procedure TFPHashList.SetCount(NewCount: Integer);begin  if (NewCount < 0) or (NewCount > MaxHashListSize)then    Error(SListCountError, NewCount);  If NewCount > FCount then    begin      If NewCount > FCapacity then        SetCapacity(NewCount);      If FCount < NewCount then        { FCapacity is NewCount rounded up to the next power of 2 }        FillChar(FHashList^[FCount], (FCapacity-FCount) div Sizeof(THashItem), 0);    end;  FCount := Newcount;end;procedure TFPHashList.SetStrCapacity(NewCapacity: Integer);{$ifdef symansistr}var  i: longint;{$endif symansistr}begin{$push}{$warnings off}  If (NewCapacity < FStrCount) or (NewCapacity > MaxHashStrSize) then     Error (SListCapacityError, NewCapacity);{$pop}  if NewCapacity = FStrCapacity then    exit;{$ifdef symansistr}{ array of ansistrings -> finalize }  if (NewCapacity < FStrCapacity) then    for i:=NewCapacity to FStrCapacity-1 do      finalize(FStrs[i]);  ReallocMem(FStrs, NewCapacity*sizeof(pansistring));  { array of ansistrings -> initialize to nil }  if (NewCapacity > FStrCapacity) then    fillchar(FStrs[FStrCapacity],(NewCapacity-FStrCapacity)*sizeof(pansistring),0);{$else symansistr}  ReallocMem(FStrs, NewCapacity);{$endif symansistr}  FStrCapacity := NewCapacity;end;procedure TFPHashList.SetHashCapacity(NewCapacity: Integer);var  power: longint;begin  If (NewCapacity < 1) then    Error (SListCapacityError, NewCapacity);  if FHashCapacity=NewCapacity then    exit;  if (NewCapacity<>0) and     not ispowerof2(NewCapacity,power) then    Error(SListCapacityPower2Error, NewCapacity);  FHashCapacity:=NewCapacity;  ReallocMem(FHashTable, FHashCapacity*sizeof(Integer));  FCapacityMask:=(1 shl power)-1;  ReHash;end;procedure TFPHashList.ReHash;var  i : Integer;begin  FillDword(FHashTable^,FHashCapacity,LongWord(-1));  For i:=0 To FCount-1 Do    AddToHashTable(i);end;constructor TFPHashList.Create;begin  SetHashCapacity(1);end;destructor TFPHashList.Destroy;begin  Clear;  if assigned(FHashTable) then    FreeMem(FHashTable);  inherited Destroy;end;function TFPHashList.AddStr(const s:TSymStr): Integer;{$ifndef symansistr}var  Len : Integer;{$endif symansistr}begin{$ifdef symansistr}  if FStrCount+1 >= FStrCapacity then    StrExpand(FStrCount+1);  FStrs[FStrCount]:=s;  result:=FStrCount;  inc(FStrCount);{$else symansistr}  len:=length(s)+1;  if FStrCount+Len >= FStrCapacity then    StrExpand(Len);  System.Move(s[0],FStrs[FStrCount],Len);  result:=FStrCount;  inc(FStrCount,Len);{$endif symansistr}end;procedure TFPHashList.AddToHashTable(Index: Integer);var  HashIndex : Integer;begin  with FHashList^[Index] do    begin      if not assigned(Data) then        exit;      HashIndex:=HashValue and FCapacityMask;      NextIndex:=FHashTable^[HashIndex];      FHashTable^[HashIndex]:=Index;    end;end;function TFPHashList.Add(const AName:TSymStr;Item: Pointer): Integer;begin  if FCount = FCapacity then    Expand;  with FHashList^[FCount] do    begin      HashValue:=FPHash(AName);      Data:=Item;      StrIndex:=AddStr(AName);    end;  AddToHashTable(FCount);  Result := FCount;  inc(FCount);end;procedure TFPHashList.Clear;begin  if Assigned(FHashList) then    begin      FCount:=0;      SetCapacity(0);      FHashList := nil;    end;  SetHashCapacity(1);  FHashTable^[0]:=-1; // sethashcapacity does not always call rehash  if Assigned(FStrs) then    begin      FStrCount:=0;      SetStrCapacity(0);      FStrs := nil;    end;end;procedure TFPHashList.Delete(Index: Integer);begin  If (Index<0) or (Index>=FCount) then    Error (SListIndexError, Index);  { Remove from HashList }  dec(FCount);  System.Move (FHashList^[Index+1], FHashList^[Index], (FCount - Index) * Sizeof(THashItem));  { All indexes are updated, we need to build the hashtable again }  Rehash;  { Shrink the list if appropriate }  if (FCapacity > 256) and (FCount < FCapacity shr 2) then    begin      FCapacity := FCapacity shr 1;      ReallocMem(FHashList, Sizeof(THashItem) * FCapacity);    end;end;function TFPHashList.Remove(Item: Pointer): Integer;begin  Result := IndexOf(Item);  If Result <> -1 then    Self.Delete(Result);end;class procedure TFPHashList.Error(const Msg: string; Data: PtrInt);{$ifndef VER2_6}noreturn;{$endif VER2_6}begin  Raise EListError.CreateFmt(Msg,[Data])  at get_caller_addr(get_frame), get_caller_frame(get_frame);end;function TFPHashList.Expand: TFPHashList;var  IncSize : Longint;begin  Result := Self;  if FCount < FCapacity then    exit;  IncSize := sizeof(ptrint)*2;  SetCapacity(FCapacity + IncSize);end;procedure TFPHashList.StrExpand(MinIncSize:Integer);var  IncSize : Longint;begin  if FStrCount+MinIncSize < FStrCapacity then    exit;  IncSize := 64;  if FStrCapacity > 255 then    Inc(IncSize, FStrCapacity shr 2);  SetStrCapacity(FStrCapacity + IncSize + MinIncSize);end;function TFPHashList.IndexOf(Item: Pointer): Integer;var  psrc  : PHashItem;  Index : integer;begin  Result:=-1;  psrc:=@FHashList^[0];  For Index:=0 To FCount-1 Do    begin      if psrc^.Data=Item then        begin          Result:=Index;          exit;        end;      inc(psrc);    end;end;function TFPHashList.InternalFind(AHash:LongWord;const AName:TSymStr;out PrevIndex:Integer):Integer;begin  prefetch(AName[1]);  Result:=FHashTable^[AHash and FCapacityMask];  PrevIndex:=-1;  while Result<>-1 do    begin      with FHashList^[Result] do        begin          if assigned(Data) and             (HashValue=AHash) and             (AName=PSymStr(@FStrs[StrIndex])^) then            exit;          PrevIndex:=Result;          Result:=NextIndex;        end;    end;end;function TFPHashList.Find(const AName:TSymStr): Pointer;var  Index,  PrevIndex : Integer;begin  Result:=nil;  Index:=InternalFind(FPHash(AName),AName,PrevIndex);  if Index=-1 then    exit;  Result:=FHashList^[Index].Data;end;function TFPHashList.FindIndexOf(const AName:TSymStr): Integer;var  PrevIndex : Integer;begin  Result:=InternalFind(FPHash(AName),AName,PrevIndex);end;function TFPHashList.FindWithHash(const AName:TSymStr;AHash:LongWord): Pointer;var  Index,  PrevIndex : Integer;begin  Result:=nil;  Index:=InternalFind(AHash,AName,PrevIndex);  if Index=-1 then    exit;  Result:=FHashList^[Index].Data;end;function TFPHashList.Rename(const AOldName,ANewName:TSymStr): Integer;var  PrevIndex,  Index : Integer;  OldHash : LongWord;begin  Result:=-1;  OldHash:=FPHash(AOldName);  Index:=InternalFind(OldHash,AOldName,PrevIndex);  if Index=-1 then    exit;  { Remove from current Hash }  if PrevIndex<>-1 then    FHashList^[PrevIndex].NextIndex:=FHashList^[Index].NextIndex  else    FHashTable^[OldHash and FCapacityMask]:=FHashList^[Index].NextIndex;  { Set new name and hash }  with FHashList^[Index] do    begin      HashValue:=FPHash(ANewName);      StrIndex:=AddStr(ANewName);    end;  { Insert back in Hash }  AddToHashTable(Index);  { Return Index }  Result:=Index;end;procedure TFPHashList.Pack;var  NewCount,  i : integer;  pdest,  psrc : PHashItem;begin  NewCount:=0;  psrc:=@FHashList^[0];  pdest:=psrc;  For I:=0 To FCount-1 Do    begin      if assigned(psrc^.Data) then        begin          pdest^:=psrc^;          inc(pdest);          inc(NewCount);        end;      inc(psrc);    end;  FCount:=NewCount;  { We need to ReHash to update the IndexNext }  ReHash;  { Release over-capacity }  SetCapacity(FCount);  SetStrCapacity(FStrCount);end;procedure TFPHashList.ShowStatistics;var  HashMean,  HashStdDev : Double;  Index,  i,j : Integer;begin  { Calculate Mean and StdDev }  HashMean:=0;  HashStdDev:=0;  for i:=0 to FHashCapacity-1 do    begin      j:=0;      Index:=FHashTable^[i];      while (Index<>-1) do        begin          inc(j);          Index:=FHashList^[Index].NextIndex;        end;      HashMean:=HashMean+j;      HashStdDev:=HashStdDev+Sqr(j);    end;  HashMean:=HashMean/FHashCapacity;  HashStdDev:=(HashStdDev-FHashCapacity*Sqr(HashMean));  If FHashCapacity>1 then    HashStdDev:=Sqrt(HashStdDev/(FHashCapacity-1))  else    HashStdDev:=0;  { Print info to stdout }  Writeln('HashSize   : ',FHashCapacity);  Writeln('HashMean   : ',HashMean:1:4);  Writeln('HashStdDev : ',HashStdDev:1:4);  Writeln('ListSize   : ',FCount,'/',FCapacity);  Writeln('StringSize : ',FStrCount,'/',FStrCapacity);end;procedure TFPHashList.ForEachCall(proc2call:TListCallback;arg:pointer);var  i : integer;  p : pointer;begin  For I:=0 To Count-1 Do    begin      p:=FHashList^[i].Data;      if assigned(p) then        proc2call(p,arg);    end;end;procedure TFPHashList.ForEachCall(proc2call:TListStaticCallback;arg:pointer);var  i : integer;  p : pointer;begin  For I:=0 To Count-1 Do    begin      p:=FHashList^[i].Data;      if assigned(p) then        proc2call(p,arg);    end;end;procedure TFPHashList.WhileEachCall(proc2call:TListCallback;arg:pointer);var  i : integer;  p : pointer;begin  i:=0;  while i<count do    begin      p:=FHashList^[i].Data;      if assigned(p) then        proc2call(p,arg);      inc(i);    end;end;procedure TFPHashList.WhileEachCall(proc2call:TListStaticCallback;arg:pointer);var  i : integer;  p : pointer;begin  i:=0;  while i<count do    begin      p:=FHashList^[i].Data;      if assigned(p) then        proc2call(p,arg);      inc(i);    end;end;{*****************************************************************************                               TFPHashObject*****************************************************************************}procedure TFPHashObject.InternalChangeOwner(HashObjectList:TFPHashObjectList;const s:TSymStr);var  Index : integer;begin  FOwner:=HashObjectList;  Index:=HashObjectList.Add(s,Self);  FStrIndex:=HashObjectList.List.List^[Index].StrIndex;end;constructor TFPHashObject.CreateNotOwned;begin  FStrIndex:=-1;end;constructor TFPHashObject.Create(HashObjectList:TFPHashObjectList;const s:TSymStr);begin  InternalChangeOwner(HashObjectList,s);end;procedure TFPHashObject.ChangeOwner(HashObjectList:TFPHashObjectList);begin  InternalChangeOwner(HashObjectList,PSymStr(@FOwner.List.Strs[FStrIndex])^);end;procedure TFPHashObject.ChangeOwnerAndName(HashObjectList:TFPHashObjectList;const s:TSymStr);begin  InternalChangeOwner(HashObjectList,s);end;procedure TFPHashObject.Rename(const ANewName:TSymStr);var  Index : integer;begin  Index:=FOwner.Rename(PSymStr(@FOwner.List.Strs[FStrIndex])^,ANewName);  if Index<>-1 then    FStrIndex:=FOwner.List.List^[Index].StrIndex;end;function TFPHashObject.GetName:TSymStr;begin  if FOwner<>nil then    Result:=PSymStr(@FOwner.List.Strs[FStrIndex])^  else    Result:='';end;function TFPHashObject.GetHash:Longword;begin  if FOwner<>nil then    Result:=FPHash(PSymStr(@FOwner.List.Strs[FStrIndex])^)  else    Result:=$ffffffff;end;{*****************************************************************************            TFPHashObjectList (Copied from rtl/objpas/classes/lists.inc)*****************************************************************************}constructor TFPHashObjectList.Create(FreeObjects : boolean = True);begin  inherited Create;  FHashList := TFPHashList.Create;  FFreeObjects := Freeobjects;end;destructor TFPHashObjectList.Destroy;begin  if (FHashList <> nil) then    begin      Clear;      FHashList.Destroy;      FHashList:=nil;    end;  inherited Destroy;end;procedure TFPHashObjectList.Clear;var  i: integer;begin  if FFreeObjects then    for i := 0 to FHashList.Count - 1 do      TObject(FHashList[i]).Free;  FHashList.Clear;end;function TFPHashObjectList.GetCount: integer;begin  Result := FHashList.Count;end;procedure TFPHashObjectList.SetCount(const AValue: integer);begin  if FHashList.Count <> AValue then    FHashList.Count := AValue;end;function TFPHashObjectList.GetItem(Index: Integer): TObject;begin  Result := TObject(FHashList[Index]);end;procedure TFPHashObjectList.SetItem(Index: Integer; AObject: TObject);begin  if OwnsObjects then    TObject(FHashList[Index]).Free;  FHashList[index] := AObject;end;procedure TFPHashObjectList.SetCapacity(NewCapacity: Integer);begin  FHashList.Capacity := NewCapacity;end;function TFPHashObjectList.GetCapacity: integer;begin  Result := FHashList.Capacity;end;function TFPHashObjectList.Add(const AName:TSymStr;AObject: TObject): Integer;begin  Result := FHashList.Add(AName,AObject);end;function TFPHashObjectList.NameOfIndex(Index: Integer): TSymStr;begin  Result := FHashList.NameOfIndex(Index);end;function TFPHashObjectList.HashOfIndex(Index: Integer): LongWord;begin  Result := FHashList.HashOfIndex(Index);end;function TFPHashObjectList.GetNextCollision(Index: Integer): Integer;begin  Result := FHashList.GetNextCollision(Index);end;procedure TFPHashObjectList.Delete(Index: Integer);begin  if OwnsObjects then    TObject(FHashList[Index]).Free;  FHashList.Delete(Index);end;function TFPHashObjectList.Expand: TFPHashObjectList;begin  FHashList.Expand;  Result := Self;end;function TFPHashObjectList.Extract(Item: TObject): TObject;begin  Result := TObject(FHashList.Extract(Item));end;function TFPHashObjectList.Remove(AObject: TObject): Integer;begin  Result := IndexOf(AObject);  if (Result <> -1) then    begin      if OwnsObjects then        TObject(FHashList[Result]).Free;      FHashList.Delete(Result);    end;end;function TFPHashObjectList.IndexOf(AObject: TObject): Integer;begin  Result := FHashList.IndexOf(Pointer(AObject));end;function TFPHashObjectList.Find(const s:TSymStr): TObject;begin  result:=TObject(FHashList.Find(s));end;function TFPHashObjectList.FindIndexOf(const s:TSymStr): Integer;begin  result:=FHashList.FindIndexOf(s);end;function TFPHashObjectList.FindWithHash(const AName:TSymStr;AHash:LongWord): Pointer;begin  Result:=TObject(FHashList.FindWithHash(AName,AHash));end;function TFPHashObjectList.Rename(const AOldName,ANewName:TSymStr): Integer;begin  Result:=FHashList.Rename(AOldName,ANewName);end;function TFPHashObjectList.FindInstanceOf(AClass: TClass; AExact: Boolean; AStartAt : Integer): Integer;var  I : Integer;begin  I:=AStartAt;  Result:=-1;  If AExact then    while (I<Count) and (Result=-1) do      If Items[i].ClassType=AClass then        Result:=I      else        Inc(I)  else    while (I<Count) and (Result=-1) do      If Items[i].InheritsFrom(AClass) then        Result:=I      else        Inc(I);end;procedure TFPHashObjectList.Pack;begin  FHashList.Pack;end;procedure TFPHashObjectList.ShowStatistics;begin  FHashList.ShowStatistics;end;procedure TFPHashObjectList.ForEachCall(proc2call:TObjectListCallback;arg:pointer);begin  FHashList.ForEachCall(TListCallBack(proc2call),arg);end;procedure TFPHashObjectList.ForEachCall(proc2call:TObjectListStaticCallback;arg:pointer);begin  FHashList.ForEachCall(TListStaticCallBack(proc2call),arg);end;procedure TFPHashObjectList.WhileEachCall(proc2call:TObjectListCallback;arg:pointer);begin  FHashList.WhileEachCall(TListCallBack(proc2call),arg);end;procedure TFPHashObjectList.WhileEachCall(proc2call:TObjectListStaticCallback;arg:pointer);begin  FHashList.WhileEachCall(TListStaticCallBack(proc2call),arg);end;{****************************************************************************                             TLinkedListItem ****************************************************************************}    constructor TLinkedListItem.Create;      begin        Previous:=nil;        Next:=nil;      end;    destructor TLinkedListItem.Destroy;      begin      end;    function TLinkedListItem.GetCopy:TLinkedListItem;      var        p : TLinkedListItem;        l : integer;      begin        p:=TLinkedListItemClass(ClassType).Create;        l:=InstanceSize;        Move(pointer(self)^,pointer(p)^,l);        Result:=p;      end;{****************************************************************************                                   TLinkedList ****************************************************************************}    constructor TLinkedList.Create;      begin        FFirst:=nil;        Flast:=nil;        FCount:=0;        FNoClear:=False;      end;    destructor TLinkedList.destroy;      begin        if not FNoClear then         Clear;      end;    function TLinkedList.empty:boolean;      begin        Empty:=(FFirst=nil);      end;    procedure TLinkedList.Insert(Item:TLinkedListItem);      begin        if FFirst=nil then         begin           FLast:=Item;           Item.Previous:=nil;           Item.Next:=nil;         end        else         begin           FFirst.Previous:=Item;           Item.Previous:=nil;           Item.Next:=FFirst;         end;        FFirst:=Item;        inc(FCount);      end;    procedure TLinkedList.InsertBefore(Item,Loc : TLinkedListItem);      begin         Item.Previous:=Loc.Previous;         Item.Next:=Loc;         Loc.Previous:=Item;         if assigned(Item.Previous) then           Item.Previous.Next:=Item         else           { if we've no next item, we've to adjust FFist }           FFirst:=Item;         inc(FCount);      end;    procedure TLinkedList.InsertAfter(Item,Loc : TLinkedListItem);      begin         Item.Next:=Loc.Next;         Loc.Next:=Item;         Item.Previous:=Loc;         if assigned(Item.Next) then           Item.Next.Previous:=Item         else           { if we've no next item, we've to adjust FLast }           FLast:=Item;         inc(FCount);      end;    procedure TLinkedList.Concat(Item:TLinkedListItem);      begin        if FFirst=nil then         begin           FFirst:=Item;           Item.Previous:=nil;           Item.Next:=nil;         end        else         begin           Flast.Next:=Item;           Item.Previous:=Flast;           Item.Next:=nil;         end;        Flast:=Item;        inc(FCount);      end;    procedure TLinkedList.remove(Item:TLinkedListItem);      begin         if Item=nil then           exit;         if (FFirst=Item) and (Flast=Item) then           begin              FFirst:=nil;              Flast:=nil;           end         else if FFirst=Item then           begin              FFirst:=Item.Next;              if assigned(FFirst) then                FFirst.Previous:=nil;           end         else if Flast=Item then           begin              Flast:=Flast.Previous;              if assigned(Flast) then                Flast.Next:=nil;           end         else           begin              Item.Previous.Next:=Item.Next;              Item.Next.Previous:=Item.Previous;           end;         Item.Next:=nil;         Item.Previous:=nil;         dec(FCount);      end;    procedure TLinkedList.clear;      var        NewNode, Next : TLinkedListItem;      begin        NewNode:=FFirst;        while assigned(NewNode) do         begin           Next:=NewNode.Next;           prefetch(next.next);           NewNode.Free;           NewNode:=Next;          end;        FLast:=nil;        FFirst:=nil;        FCount:=0;      end;    function TLinkedList.GetFirst:TLinkedListItem;      begin         if FFirst=nil then          GetFirst:=nil         else          begin            GetFirst:=FFirst;            if FFirst=FLast then             FLast:=nil;            FFirst:=FFirst.Next;            dec(FCount);          end;      end;    function TLinkedList.GetLast:TLinkedListItem;      begin         if FLast=nil then          Getlast:=nil         else          begin            Getlast:=FLast;            if FLast=FFirst then             FFirst:=nil;            FLast:=FLast.Previous;            dec(FCount);          end;      end;    procedure TLinkedList.insertList(p : TLinkedList);      begin         { empty List ? }         if (p.FFirst=nil) then           exit;         p.Flast.Next:=FFirst;         { we have a double Linked List }         if assigned(FFirst) then           FFirst.Previous:=p.Flast;         FFirst:=p.FFirst;         if (FLast=nil) then           Flast:=p.Flast;         inc(FCount,p.FCount);         { p becomes empty }         p.FFirst:=nil;         p.Flast:=nil;         p.FCount:=0;      end;    procedure TLinkedList.insertListBefore(Item:TLinkedListItem;p : TLinkedList);      begin         { empty List ? }         if (p.FFirst=nil) then           exit;         if (Item=nil) then           begin             { Insert at begin }             InsertList(p);             exit;           end         else           begin             p.FLast.Next:=Item;             p.FFirst.Previous:=Item.Previous;             if assigned(Item.Previous) then               Item.Previous.Next:=p.FFirst             else               FFirst:=p.FFirst;             Item.Previous:=p.FLast;             inc(FCount,p.FCount);           end;         { p becomes empty }         p.FFirst:=nil;         p.Flast:=nil;         p.FCount:=0;      end;    procedure TLinkedList.insertListAfter(Item:TLinkedListItem;p : TLinkedList);      begin         { empty List ? }         if (p.FFirst=nil) then           exit;         if (Item=nil) then           begin             { Insert at begin }             InsertList(p);             exit;           end         else           begin             p.FFirst.Previous:=Item;             p.FLast.Next:=Item.Next;             if assigned(Item.Next) then               Item.Next.Previous:=p.FLast             else               FLast:=p.FLast;             Item.Next:=p.FFirst;             inc(FCount,p.FCount);           end;         { p becomes empty }         p.FFirst:=nil;         p.Flast:=nil;         p.FCount:=0;      end;    procedure TLinkedList.concatList(p : TLinkedList);      begin        if (p.FFirst=nil) then         exit;        if FFirst=nil then         FFirst:=p.FFirst        else         begin           FLast.Next:=p.FFirst;           p.FFirst.Previous:=Flast;         end;        Flast:=p.Flast;        inc(FCount,p.FCount);        { make p empty }        p.Flast:=nil;        p.FFirst:=nil;        p.FCount:=0;      end;    procedure TLinkedList.insertListcopy(p : TLinkedList);      var        NewNode,NewNode2 : TLinkedListItem;      begin        NewNode:=p.Last;        while assigned(NewNode) do         begin           NewNode2:=NewNode.Getcopy;           if assigned(NewNode2) then            Insert(NewNode2);           NewNode:=NewNode.Previous;         end;      end;    procedure TLinkedList.concatListcopy(p : TLinkedList);      var        NewNode,NewNode2 : TLinkedListItem;      begin        NewNode:=p.First;        while assigned(NewNode) do         begin           NewNode2:=NewNode.Getcopy;           if assigned(NewNode2) then            Concat(NewNode2);           NewNode:=NewNode.Next;         end;      end;    procedure TLinkedList.RemoveAll;      begin        FFirst:=nil;        FLast:=nil;        FCount:=0;      end;{****************************************************************************                             TCmdStrListItem ****************************************************************************}    constructor TCmdStrListItem.Create(const s:TCmdStr);      begin        inherited Create;        FPStr:=s;      end;    destructor TCmdStrListItem.Destroy;      begin        FPStr:='';      end;    function TCmdStrListItem.Str:TCmdStr;      begin        Str:=FPStr;      end;    function TCmdStrListItem.GetCopy:TLinkedListItem;      begin        Result:=(inherited GetCopy);        { TLinkedListItem.GetCopy performs a "move" to copy all data -> reinit          the ansistring, so the refcount is properly increased }        Initialize(TCmdStrListItem(Result).FPStr);        TCmdStrListItem(Result).FPStr:=FPstr;      end;{****************************************************************************                           TCmdStrList ****************************************************************************}    constructor TCmdStrList.Create;      begin         inherited Create;         FDoubles:=true;      end;    constructor TCmdStrList.Create_no_double;      begin         inherited Create;         FDoubles:=false;      end;    procedure TCmdStrList.insert(const s : TCmdStr);      begin         if (s='') or            ((not FDoubles) and (findcase(s)<>nil)) then          exit;         inherited insert(TCmdStrListItem.create(s));      end;    procedure TCmdStrList.concat(const s : TCmdStr);      begin         if (s='') or            ((not FDoubles) and (findcase(s)<>nil)) then          exit;         inherited concat(TCmdStrListItem.create(s));      end;    procedure TCmdStrList.remove(const s : TCmdStr);      var        p : TCmdStrListItem;      begin        if s='' then         exit;        p:=findcase(s);        if assigned(p) then         begin           inherited Remove(p);           p.Free;         end;      end;    function TCmdStrList.GetFirst : TCmdStr;      var         p : TCmdStrListItem;      begin         p:=TCmdStrListItem(inherited GetFirst);         if p=nil then          GetFirst:=''         else          begin            GetFirst:=p.FPStr;            p.free;          end;      end;    function TCmdStrList.Getlast : TCmdStr;      var         p : TCmdStrListItem;      begin         p:=TCmdStrListItem(inherited Getlast);         if p=nil then          Getlast:=''         else          begin            Getlast:=p.FPStr;            p.free;          end;      end;    function TCmdStrList.FindCase(const s:TCmdStr):TCmdStrListItem;      var        NewNode : TCmdStrListItem;      begin        result:=nil;        if s='' then         exit;        NewNode:=TCmdStrListItem(FFirst);        while assigned(NewNode) do         begin           if NewNode.FPStr=s then            begin              result:=NewNode;              exit;            end;           NewNode:=TCmdStrListItem(NewNode.Next);         end;      end;    function TCmdStrList.Find(const s:TCmdStr):TCmdStrListItem;      var        NewNode : TCmdStrListItem;      begin        result:=nil;        if s='' then         exit;        NewNode:=TCmdStrListItem(FFirst);        while assigned(NewNode) do         begin           if SysUtils.CompareText(s, NewNode.FPStr)=0 then            begin              result:=NewNode;              exit;            end;           NewNode:=TCmdStrListItem(NewNode.Next);         end;      end;    procedure TCmdStrList.InsertItem(item:TCmdStrListItem);      begin        inherited Insert(item);      end;    procedure TCmdStrList.ConcatItem(item:TCmdStrListItem);      begin        inherited Concat(item);      end;{****************************************************************************                                tdynamicarray****************************************************************************}    constructor tdynamicarray.create(Ablocksize:longword);      begin        FPosn:=0;        FPosnblock:=nil;        FFirstblock:=nil;        FLastblock:=nil;        FCurrBlockSize:=0;        { Every block needs at least a header and alignment slack,          therefore its size cannot be arbitrarily small. However,          the blocksize argument is often confused with data size.          See e.g. Mantis #20929. }        if Ablocksize<mindynamicblocksize then          Ablocksize:=mindynamicblocksize;        FMaxBlockSize:=Ablocksize;        grow;      end;    destructor tdynamicarray.destroy;      var        hp : pdynamicblock;      begin        while assigned(FFirstblock) do         begin           hp:=FFirstblock;           FFirstblock:=FFirstblock^.Next;           Freemem(hp);         end;      end;    function  tdynamicarray.size:longword;      begin        if assigned(FLastblock) then         size:=FLastblock^.pos+FLastblock^.used        else         size:=0;      end;    procedure tdynamicarray.reset;      var        hp : pdynamicblock;      begin        while assigned(FFirstblock) do         begin           hp:=FFirstblock;           FFirstblock:=FFirstblock^.Next;           Freemem(hp);         end;        FPosn:=0;        FPosnblock:=nil;        FFirstblock:=nil;        FLastblock:=nil;        grow;      end;    procedure tdynamicarray.grow;      var        nblock  : pdynamicblock;        OptBlockSize,        IncSize : integer;      begin        if CurrBlockSize<FMaxBlocksize then          begin            IncSize := mindynamicblocksize;            if FCurrBlockSize > 255 then              Inc(IncSize, FCurrBlockSize shr 2);            inc(FCurrBlockSize,IncSize);          end;        if CurrBlockSize>FMaxBlocksize then          FCurrBlockSize:=FMaxBlocksize;        { Calculate the most optimal size so there is no alignment overhead          lost in the heap manager }        OptBlockSize:=cutils.Align(CurrBlockSize+dynamicblockbasesize,16)-dynamicblockbasesize-sizeof(ptrint);        Getmem(nblock,OptBlockSize+dynamicblockbasesize);        if not assigned(FFirstblock) then         begin           FFirstblock:=nblock;           FPosnblock:=nblock;           nblock^.pos:=0;         end        else         begin           FLastblock^.Next:=nblock;           nblock^.pos:=FLastblock^.pos+FLastblock^.size;         end;        nblock^.used:=0;        nblock^.size:=OptBlockSize;        nblock^.Next:=nil;        fillchar(nblock^.data,nblock^.size,0);        FLastblock:=nblock;      end;    procedure tdynamicarray.align(i:longword);      var        j : longword;      begin        j:=(FPosn mod i);        if j<>0 then         begin           j:=i-j;           if FPosnblock^.used+j>FPosnblock^.size then            begin              dec(j,FPosnblock^.size-FPosnblock^.used);              FPosnblock^.used:=FPosnblock^.size;              grow;              FPosnblock:=FLastblock;            end;           inc(FPosnblock^.used,j);           inc(FPosn,j);         end;      end;    procedure tdynamicarray.seek(i:longword);      begin        if (i<FPosnblock^.pos) or (i>=FPosnblock^.pos+FPosnblock^.size) then         begin           { set FPosnblock correct if the size is bigger then             the current block }           if FPosnblock^.pos>i then            FPosnblock:=FFirstblock;           while assigned(FPosnblock) do            begin              if FPosnblock^.pos+FPosnblock^.size>i then               break;              FPosnblock:=FPosnblock^.Next;            end;           { not found ? then increase blocks }           if not assigned(FPosnblock) then            begin              repeat                { the current FLastblock is now also fully used }                FLastblock^.used:=FLastblock^.size;                grow;                FPosnblock:=FLastblock;              until FPosnblock^.pos+FPosnblock^.size>=i;            end;         end;        FPosn:=i;        if FPosn-FPosnblock^.pos>FPosnblock^.used then         FPosnblock^.used:=FPosn-FPosnblock^.pos;      end;    procedure tdynamicarray.write(const d;len:longword);      var        p : pchar;        i,j : longword;      begin        p:=pchar(@d);        while (len>0) do         begin           i:=FPosn-FPosnblock^.pos;           if i+len>=FPosnblock^.size then            begin              j:=FPosnblock^.size-i;              move(p^,FPosnblock^.data[i],j);              inc(p,j);              inc(FPosn,j);              dec(len,j);              FPosnblock^.used:=FPosnblock^.size;              if assigned(FPosnblock^.Next) then               FPosnblock:=FPosnblock^.Next              else               begin                 grow;                 FPosnblock:=FLastblock;               end;            end           else            begin              move(p^,FPosnblock^.data[i],len);              inc(p,len);              inc(FPosn,len);              i:=FPosn-FPosnblock^.pos;              if i>FPosnblock^.used then               FPosnblock^.used:=i;              len:=0;            end;         end;      end;    procedure tdynamicarray.writestr(const s:string);      begin        write(s[1],length(s));      end;    function tdynamicarray.read(var d;len:longword):longword;      var        p : pchar;        i,j,res : longword;      begin        res:=0;        p:=pchar(@d);        while (len>0) do         begin           i:=FPosn-FPosnblock^.pos;           if i+len>=FPosnblock^.used then            begin              j:=FPosnblock^.used-i;              move(FPosnblock^.data[i],p^,j);              inc(p,j);              inc(FPosn,j);              inc(res,j);              dec(len,j);              if assigned(FPosnblock^.Next) then               FPosnblock:=FPosnblock^.Next              else               break;            end           else            begin              move(FPosnblock^.data[i],p^,len);              inc(p,len);              inc(FPosn,len);              inc(res,len);              len:=0;            end;         end;        read:=res;      end;    procedure tdynamicarray.readstream(f:TCStream;maxlen:longword);      var        i,left : longword;      begin        repeat          left:=FPosnblock^.size-FPosnblock^.used;          if left>maxlen then           left:=maxlen;          i:=f.Read(FPosnblock^.data[FPosnblock^.used],left);          dec(maxlen,i);          inc(FPosnblock^.used,i);          if FPosnblock^.used=FPosnblock^.size then           begin             if assigned(FPosnblock^.Next) then              FPosnblock:=FPosnblock^.Next             else              begin                grow;                FPosnblock:=FLastblock;              end;           end;        until (i<left) or (maxlen=0);      end;    procedure tdynamicarray.writestream(f:TCStream);      var        hp : pdynamicblock;      begin        hp:=FFirstblock;        while assigned(hp) do         begin           f.Write(hp^.data,hp^.used);           hp:=hp^.Next;         end;      end;    function tdynamicarray.equal(other:tdynamicarray):boolean;      begin        result:=false;        { TODO }      end;{****************************************************************************                                thashset****************************************************************************}    constructor THashSet.Create(InitSize: Integer; OwnKeys, OwnObjects: Boolean);      var        I: Integer;      begin        inherited Create;        FOwnsObjects := OwnObjects;        FOwnsKeys := OwnKeys;        I := 64;        while I < InitSize do I := I shl 1;        FBucketCount := I;        FBucket := AllocMem(I * sizeof(PHashSetItem));      end;    destructor THashSet.Destroy;      begin        Clear;        FreeMem(FBucket);        inherited Destroy;      end;    procedure THashSet.Clear;      var        I: Integer;        item, next: PHashSetItem;      begin        for I := 0 to FBucketCount-1 do        begin          item := FBucket[I];          while Assigned(item) do          begin            next := item^.Next;            if FOwnsObjects then              item^.Data.Free;            if FOwnsKeys then              FreeMem(item^.Key);            FreeItem(item);            item := next;          end;        end;        FillChar(FBucket^, FBucketCount * sizeof(PHashSetItem), 0);      end;    function THashSet.Find(Key: Pointer; KeyLen: Integer): PHashSetItem;      var        Dummy: Boolean;      begin        Result := Lookup(Key, KeyLen, Dummy, False);      end;    function THashSet.FindOrAdd(Key: Pointer; KeyLen: Integer;        var Found: Boolean): PHashSetItem;      begin        Result := Lookup(Key, KeyLen, Found, True);      end;    function THashSet.FindOrAdd(Key: Pointer; KeyLen: Integer): PHashSetItem;      var        Dummy: Boolean;      begin        Result := Lookup(Key, KeyLen, Dummy, True);      end;    function THashSet.Get(Key: Pointer; KeyLen: Integer): TObject;      var        e: PHashSetItem;        Dummy: Boolean;      begin        e := Lookup(Key, KeyLen, Dummy, False);        if Assigned(e) then          Result := e^.Data        else          Result := nil;      end;    function THashSet.Lookup(Key: Pointer; KeyLen: Integer;      var Found: Boolean; CanCreate: Boolean): PHashSetItem;      var        Entry: PPHashSetItem;        h: LongWord;      begin        h := FPHash(Key, KeyLen);        Entry := @FBucket[h and (FBucketCount-1)];        while Assigned(Entry^) and          not ((Entry^^.HashValue = h) and (Entry^^.KeyLength = KeyLen) and            (CompareByte(Entry^^.Key^, Key^, KeyLen) = 0)) do              Entry := @Entry^^.Next;        Found := Assigned(Entry^);        if Found or (not CanCreate) then          begin            Result := Entry^;            Exit;          end;        if FCount > FBucketCount then  { arbitrary limit, probably too high }          begin            { rehash and repeat search }            Resize(FBucketCount * 2);            Result := Lookup(Key, KeyLen, Found, CanCreate);          end        else          begin            GetMem(Result,SizeOfItem);            if FOwnsKeys then            begin              GetMem(Result^.Key, KeyLen);              Move(Key^, Result^.Key^, KeyLen);            end            else              Result^.Key := Key;            Result^.KeyLength := KeyLen;            Result^.HashValue := h;            Result^.Data := nil;            Result^.Next := nil;            Inc(FCount);            Entry^ := Result;          end;        end;    procedure THashSet.Resize(NewCapacity: LongWord);      var        p, chain: PPHashSetItem;        i: Integer;        e, n: PHashSetItem;      begin        p := AllocMem(NewCapacity * SizeOf(PHashSetItem));        for i := 0 to FBucketCount-1 do          begin            e := FBucket[i];            while Assigned(e) do            begin              chain := @p[e^.HashValue and (NewCapacity-1)];              n := e^.Next;              e^.Next := chain^;              chain^ := e;              e := n;            end;          end;        FBucketCount := NewCapacity;        FreeMem(FBucket);        FBucket := p;      end;    class procedure THashSet.FreeItem(item: PHashSetItem);      begin        Dispose(item);      end;    class function THashSet.SizeOfItem: Integer;      begin        Result := SizeOf(THashSetItem);      end;    function THashSet.Remove(Entry: PHashSetItem): Boolean;      var        chain: PPHashSetItem;      begin        chain := @FBucket[Entry^.HashValue mod FBucketCount];        while Assigned(chain^) do          begin            if chain^ = Entry then              begin                chain^ := Entry^.Next;                if FOwnsObjects then                  Entry^.Data.Free;                if FOwnsKeys then                  FreeMem(Entry^.Key);                FreeItem(Entry);                Dec(FCount);                Result := True;                Exit;              end;            chain := @chain^^.Next;          end;        Result := False;      end;{****************************************************************************                                ttaghashset****************************************************************************}    function TTagHashSet.Lookup(Key: Pointer; KeyLen: Integer;      Tag: LongWord; var Found: Boolean; CanCreate: Boolean): PTagHashSetItem;      var        Entry: PPTagHashSetItem;        h: LongWord;      begin        h := FPHash(Key, KeyLen, Tag);        Entry := @PPTagHashSetItem(FBucket)[h and (FBucketCount-1)];        while Assigned(Entry^) and          not ((Entry^^.HashValue = h) and (Entry^^.KeyLength = KeyLen) and            (Entry^^.Tag = Tag) and (CompareByte(Entry^^.Key^, Key^, KeyLen) = 0)) do              Entry := @Entry^^.Next;        Found := Assigned(Entry^);        if Found or (not CanCreate) then          begin            Result := Entry^;            Exit;          end;        if FCount > FBucketCount then  { arbitrary limit, probably too high }          begin            { rehash and repeat search }            Resize(FBucketCount * 2);            Result := Lookup(Key, KeyLen, Tag, Found, CanCreate);          end        else          begin            Getmem(Result,SizeOfItem);            if FOwnsKeys then            begin              GetMem(Result^.Key, KeyLen);              Move(Key^, Result^.Key^, KeyLen);            end            else              Result^.Key := Key;            Result^.KeyLength := KeyLen;            Result^.HashValue := h;            Result^.Tag := Tag;            Result^.Data := nil;            Result^.Next := nil;            Inc(FCount);            Entry^ := Result;          end;      end;    class procedure TTagHashSet.FreeItem(item: PHashSetItem);      begin        Dispose(PTagHashSetItem(item));      end;    class function TTagHashSet.SizeOfItem: Integer;      begin        Result := SizeOf(TTagHashSetItem);      end;    function TTagHashSet.Find(Key: Pointer; KeyLen: Integer; Tag: LongWord): PTagHashSetItem;      var        Dummy: Boolean;      begin        Result := Lookup(Key, KeyLen, Tag, Dummy, False);      end;    function TTagHashSet.FindOrAdd(Key: Pointer; KeyLen: Integer; Tag: LongWord;      var Found: Boolean): PTagHashSetItem;      begin        Result := Lookup(Key, KeyLen, Tag, Found, True);      end;    function TTagHashSet.FindOrAdd(Key: Pointer; KeyLen: Integer; Tag: LongWord): PTagHashSetItem;      var        Dummy: Boolean;      begin        Result := Lookup(Key, KeyLen, Tag, Dummy, True);      end;    function TTagHashSet.Get(Key: Pointer; KeyLen: Integer; Tag: LongWord): TObject;      var        e: PTagHashSetItem;        Dummy: Boolean;      begin        e := Lookup(Key, KeyLen, Tag, Dummy, False);        if Assigned(e) then          Result := e^.Data        else          Result := nil;      end;{****************************************************************************                                tbitset****************************************************************************}    constructor tbitset.create(initsize: longint);      begin        create_bytesize((initsize+7) div 8);      end;    constructor tbitset.create_bytesize(bytesize: longint);      begin        fdatasize:=bytesize;        getmem(fdata,fdataSize);        clear;      end;    destructor tbitset.destroy;      begin        freemem(fdata,fdatasize);        inherited destroy;      end;    procedure tbitset.clear;      begin        fillchar(fdata^,fdatasize,0);      end;    procedure tbitset.grow(nsize: longint);      begin        reallocmem(fdata,nsize);        fillchar(fdata[fdatasize],nsize-fdatasize,0);        fdatasize:=nsize;      end;    procedure tbitset.include(index: longint);      var        dataindex: longint;      begin        { don't use bitpacked array, not endian-safe }        dataindex:=index shr 3;        if (dataindex>=datasize) then          grow(dataindex+16);        fdata[dataindex]:=fdata[dataindex] or (1 shl (index and 7));      end;    procedure tbitset.exclude(index: longint);      var        dataindex: longint;      begin        dataindex:=index shr 3;        if (dataindex>=datasize) then          exit;        fdata[dataindex]:=fdata[dataindex] and not(1 shl (index and 7));      end;    function tbitset.isset(index: longint): boolean;      var        dataindex: longint;      begin        dataindex:=index shr 3;        result:=          (dataindex<datasize) and          (((fdata[dataindex] shr (index and 7)) and 1)<>0);      end;    procedure tbitset.addset(aset: tbitset);      var        i: longint;      begin        if (aset.datasize>datasize) then          grow(aset.datasize);        for i:=0 to aset.datasize-1 do          fdata[i]:=fdata[i] or aset.data[i];      end;    procedure tbitset.subset(aset: tbitset);      var        i: longint;      begin        for i:=0 to min(datasize,aset.datasize)-1 do          fdata[i]:=fdata[i] and not(aset.data[i]);      end;end.
 |