1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935 |
- {
- $Id$
- This file is part of the Free Pascal run time library.
- Copyright (c) 1999-2000 by the Free Pascal development team.
- Objects.pas clone for Free Pascal
- See the file COPYING.FPC, included in this distribution,
- for details about the copyright.
- 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.
- **********************************************************************}
- {************[ SOURCE FILE OF FREE VISION ]****************}
- { }
- { System independent clone of objects.pas }
- { }
- { Interface Copyright (c) 1992 Borland International }
- { }
- { Parts Copyright (c) 1999-2000 by Florian Klaempfl }
- { [email protected] }
- { }
- { Parts Copyright (c) 1999-2000 by Frank ZAGO }
- { [email protected] }
- { }
- { Parts Copyright (c) 1999-2000 by MH Spiegel }
- { }
- { Parts Copyright (c) 1996, 1999-2000 by Leon de Boer }
- { [email protected] }
- { }
- { Free Vision project coordinator Balazs Scheidler }
- { [email protected] }
- { }
- UNIT Objects;
- {<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>}
- INTERFACE
- {<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>}
- {==== Select assembler ==============================================}
- {$IFDEF CPU86}
- {$ASMMODE ATT}
- {$ENDIF}
- {==== Compiler directives ===========================================}
- {$H-} { No ansistrings }
- {$E+} { Emulation is on }
- {$X+} { Extended syntax is ok }
- {$R-} { Disable range checking }
- {$ifndef Unix}
- {$S-} { Disable Stack Checking }
- {$endif}
- {$I-} { Disable IO Checking }
- {$Q-} { Disable Overflow Checking }
- {$V-} { Turn off strict VAR strings }
- {====================================================================}
- {***************************************************************************}
- { PUBLIC CONSTANTS }
- {***************************************************************************}
- {---------------------------------------------------------------------------}
- { STREAM ERROR STATE MASKS }
- {---------------------------------------------------------------------------}
- CONST
- stOk = 0; { No stream error }
- stError = -1; { Access error }
- stInitError = -2; { Initialize error }
- stReadError = -3; { Stream read error }
- stWriteError = -4; { Stream write error }
- stGetError = -5; { Get object error }
- stPutError = -6; { Put object error }
- stSeekError = -7; { Seek error in stream }
- stOpenError = -8; { Error opening stream }
- {---------------------------------------------------------------------------}
- { STREAM ACCESS MODE CONSTANTS }
- {---------------------------------------------------------------------------}
- CONST
- stCreate = $3C00; { Create new file }
- stOpenRead = $3D00; { Read access only }
- stOpenWrite = $3D01; { Write access only }
- stOpen = $3D02; { Read/write access }
- {---------------------------------------------------------------------------}
- { TCollection ERROR CODES }
- {---------------------------------------------------------------------------}
- CONST
- coIndexError = -1; { Index out of range }
- coOverflow = -2; { Overflow }
- {---------------------------------------------------------------------------}
- { VMT HEADER CONSTANT - HOPEFULLY WE CAN DROP THIS LATER }
- {---------------------------------------------------------------------------}
- CONST
- vmtHeaderSize = 8; { VMT header size }
- CONST
- {---------------------------------------------------------------------------}
- { MAXIUM DATA SIZES }
- {---------------------------------------------------------------------------}
- {$IFDEF FPC}
- MaxBytes = 128*1024*128; { Maximum data size }
- {$ELSE}
- MaxBytes = 16384;
- {$ENDIF}
- MaxWords = MaxBytes DIV SizeOf(Word); { Max word data size }
- MaxPtrs = MaxBytes DIV SizeOf(Pointer); { Max ptr data size }
- MaxCollectionSize = MaxBytes DIV SizeOf(Pointer); { Max collection size }
- MaxTPCompatibleCollectionSize = 65520 div 4;
- {***************************************************************************}
- { PUBLIC TYPE DEFINITIONS }
- {***************************************************************************}
- {---------------------------------------------------------------------------}
- { CHARACTER SET }
- {---------------------------------------------------------------------------}
- TYPE
- TCharSet = SET Of Char; { Character set }
- PCharSet = ^TCharSet; { Character set ptr }
- {---------------------------------------------------------------------------}
- { GENERAL ARRAYS }
- {---------------------------------------------------------------------------}
- TYPE
- TByteArray = ARRAY [0..MaxBytes-1] Of Byte; { Byte array }
- PByteArray = ^TByteArray; { Byte array pointer }
- TWordArray = ARRAY [0..MaxWords-1] Of Word; { Word array }
- PWordArray = ^TWordArray; { Word array pointer }
- TPointerArray = Array [0..MaxPtrs-1] Of Pointer; { Pointer array }
- PPointerArray = ^TPointerArray; { Pointer array ptr }
- {---------------------------------------------------------------------------}
- { POINTER TO STRING }
- {---------------------------------------------------------------------------}
- TYPE
- PString = PShortString; { String pointer }
- {---------------------------------------------------------------------------}
- { OS dependent File type / consts }
- {---------------------------------------------------------------------------}
- type
- FNameStr = String;
- THandle = longint;
- const
- MaxReadBytes = $7fffffff;
- invalidhandle = -1;
- {---------------------------------------------------------------------------}
- { DOS ASCIIZ FILENAME }
- {---------------------------------------------------------------------------}
- TYPE
- AsciiZ = Array [0..255] Of Char; { Filename array }
- {---------------------------------------------------------------------------}
- { BIT SWITCHED TYPE CONSTANTS }
- {---------------------------------------------------------------------------}
- TYPE
- Sw_Word = Cardinal; { Long Word now }
- Sw_Integer = LongInt; { Long integer now }
- {***************************************************************************}
- { PUBLIC RECORD DEFINITIONS }
- {***************************************************************************}
- {---------------------------------------------------------------------------}
- { TYPE CONVERSION RECORDS }
- {---------------------------------------------------------------------------}
- TYPE
- WordRec = packed RECORD
- {$ifdef ENDIAN_LITTLE}
- Lo, Hi: Byte; { Word to bytes }
- {$else}
- Hi,Lo: Byte;
- {$endif}
- END;
- LongRec = packed RECORD
- {$ifdef ENDIAN_LITTLE}
- Lo, Hi: Word; { LongInt to words }
- {$else}
- Hi,Lo: Word; { LongInt to words }
- {$endif}
- END;
- PtrRec = packed RECORD
- Ofs, Seg: Word; { Pointer to words }
- END;
- {---------------------------------------------------------------------------}
- { TStreamRec RECORD - STREAM OBJECT RECORD }
- {---------------------------------------------------------------------------}
- TYPE
- PStreamRec = ^TStreamRec; { Stream record ptr }
- TStreamRec = Packed RECORD
- ObjType: Sw_Word; { Object type id }
- VmtLink: pointer; { VMT link }
- Load : Pointer; { Object load code }
- Store: Pointer; { Object store code }
- Next : PStreamRec; { Next stream record }
- END;
- {***************************************************************************}
- { PUBLIC OBJECT DEFINITIONS }
- {***************************************************************************}
- {---------------------------------------------------------------------------}
- { TPoint OBJECT - POINT OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- PPoint = ^TPoint;
- TPoint = OBJECT
- X, Y: Sw_Integer;
- END;
- {---------------------------------------------------------------------------}
- { TRect OBJECT - RECTANGLE OBJECT }
- {---------------------------------------------------------------------------}
- PRect = ^TRect;
- TRect = OBJECT
- A, B: TPoint; { Corner points }
- FUNCTION Empty: Boolean;
- FUNCTION Equals (R: TRect): Boolean;
- FUNCTION Contains (P: TPoint): Boolean;
- PROCEDURE Copy (R: TRect);
- PROCEDURE Union (R: TRect);
- PROCEDURE Intersect (R: TRect);
- PROCEDURE Move (ADX, ADY: Sw_Integer);
- PROCEDURE Grow (ADX, ADY: Sw_Integer);
- PROCEDURE Assign (XA, YA, XB, YB: Sw_Integer);
- END;
- {---------------------------------------------------------------------------}
- { TObject OBJECT - BASE ANCESTOR OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- TObject = OBJECT
- CONSTRUCTOR Init;
- PROCEDURE Free;
- FUNCTION Is_Object(P:Pointer):Boolean;
- DESTRUCTOR Done; Virtual;
- END;
- PObject = ^TObject;
- { ******************************* REMARK ****************************** }
- { Two new virtual methods have been added to the object in the form of }
- { Close and Open. The main use here is in the Disk Based Descendants }
- { the calls open and close the given file so these objects can be }
- { used like standard files. Two new fields have also been added to }
- { speed up seeks on descendants. All existing code will compile and }
- { work completely normally oblivious to these new methods and fields. }
- { ****************************** END REMARK *** Leon de Boer, 15May96 * }
- {---------------------------------------------------------------------------}
- { TStream OBJECT - STREAM ANCESTOR OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- TStream = OBJECT (TObject)
- Status : Integer; { Stream status }
- ErrorInfo : Integer; { Stream error info }
- StreamSize: LongInt; { Stream current size }
- Position : LongInt; { Current position }
- TPCompatible : Boolean;
- CONSTRUCTOR Init;
- FUNCTION Get: PObject;
- FUNCTION StrRead: PChar;
- FUNCTION GetPos: Longint; Virtual;
- FUNCTION GetSize: Longint; Virtual;
- FUNCTION ReadStr: PString;
- PROCEDURE Open (OpenMode: Word); Virtual;
- PROCEDURE Close; Virtual;
- PROCEDURE Reset;
- PROCEDURE Flush; Virtual;
- PROCEDURE Truncate; Virtual;
- PROCEDURE Put (P: PObject);
- PROCEDURE StrWrite (P: PChar);
- PROCEDURE WriteStr (P: PString);
- PROCEDURE Seek (Pos: LongInt); Virtual;
- PROCEDURE Error (Code, Info: Integer); Virtual;
- PROCEDURE Read (Var Buf; Count: Sw_Word); Virtual;
- PROCEDURE Write (Var Buf; Count: Sw_Word); Virtual;
- PROCEDURE CopyFrom (Var S: TStream; Count: Longint);
- END;
- PStream = ^TStream;
- { ******************************* REMARK ****************************** }
- { A few minor changes to this object and an extra field added called }
- { FName which holds an AsciiZ array of the filename this allows the }
- { streams file to be opened and closed like a normal text file. All }
- { existing code should work without any changes. }
- { ****************************** END REMARK *** Leon de Boer, 19May96 * }
- {---------------------------------------------------------------------------}
- { TDosStream OBJECT - DOS FILE STREAM OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- TDosStream = OBJECT (TStream)
- Handle: THandle; { DOS file handle }
- FName : AsciiZ; { AsciiZ filename }
- CONSTRUCTOR Init (FileName: FNameStr; Mode: Word);
- DESTRUCTOR Done; Virtual;
- PROCEDURE Close; Virtual;
- PROCEDURE Truncate; Virtual;
- PROCEDURE Seek (Pos: LongInt); Virtual;
- PROCEDURE Open (OpenMode: Word); Virtual;
- PROCEDURE Read (Var Buf; Count: Sw_Word); Virtual;
- PROCEDURE Write (Var Buf; Count: Sw_Word); Virtual;
- private
- FileInfo : File;
- END;
- PDosStream = ^TDosStream;
- { ******************************* REMARK ****************************** }
- { A few minor changes to this object and an extra field added called }
- { lastmode which holds the read or write condition last using the }
- { speed up buffer which helps speed up the flush, position and size }
- { functions. All existing code should work without any changes. }
- { ****************************** END REMARK *** Leon de Boer, 19May96 * }
- {---------------------------------------------------------------------------}
- { TBufStream OBJECT - BUFFERED DOS FILE STREAM }
- {---------------------------------------------------------------------------}
- TYPE
- TBufStream = OBJECT (TDosStream)
- LastMode: Byte; { Last buffer mode }
- BufSize : Sw_Word; { Buffer size }
- BufPtr : Sw_Word; { Buffer start }
- BufEnd : Sw_Word; { Buffer end }
- Buffer : PByteArray; { Buffer allocated }
- CONSTRUCTOR Init (FileName: FNameStr; Mode, Size: Word);
- DESTRUCTOR Done; Virtual;
- PROCEDURE Close; Virtual;
- PROCEDURE Flush; Virtual;
- PROCEDURE Truncate; Virtual;
- PROCEDURE Seek (Pos: LongInt); Virtual;
- PROCEDURE Open (OpenMode: Word); Virtual;
- PROCEDURE Read (Var Buf; Count: Sw_Word); Virtual;
- PROCEDURE Write (Var Buf; Count: Sw_Word); Virtual;
- END;
- PBufStream = ^TBufStream;
- { ******************************* REMARK ****************************** }
- { All the changes here should be completely transparent to existing }
- { code. Basically the memory blocks do not have to be base segments }
- { but this means our list becomes memory blocks rather than segments. }
- { The stream will also expand like the other standard streams!! }
- { ****************************** END REMARK *** Leon de Boer, 19May96 * }
- {---------------------------------------------------------------------------}
- { TMemoryStream OBJECT - MEMORY STREAM OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- TMemoryStream = OBJECT (TStream)
- BlkCount: Sw_Word; { Number of segments }
- BlkSize : Word; { Memory block size }
- MemSize : LongInt; { Memory alloc size }
- BlkList : PPointerArray; { Memory block list }
- CONSTRUCTOR Init (ALimit: Longint; ABlockSize: Word);
- DESTRUCTOR Done; Virtual;
- PROCEDURE Truncate; Virtual;
- PROCEDURE Read (Var Buf; Count: Sw_Word); Virtual;
- PROCEDURE Write (Var Buf; Count: Sw_Word); Virtual;
- PRIVATE
- FUNCTION ChangeListSize (ALimit: Sw_Word): Boolean;
- END;
- PMemoryStream = ^TMemoryStream;
- TYPE
- TItemList = Array [0..MaxCollectionSize - 1] Of Pointer;
- PItemList = ^TItemList;
- { ******************************* REMARK ****************************** }
- { The changes here look worse than they are. The Sw_Integer simply }
- { switches between Integers and LongInts if switched between 16 and 32 }
- { bit code. All existing code will compile without any changes. }
- { ****************************** END REMARK *** Leon de Boer, 10May96 * }
- {---------------------------------------------------------------------------}
- { TCollection OBJECT - COLLECTION ANCESTOR OBJECT }
- {---------------------------------------------------------------------------}
- TCollection = OBJECT (TObject)
- Items: PItemList; { Item list pointer }
- Count: Sw_Integer; { Item count }
- Limit: Sw_Integer; { Item limit count }
- Delta: Sw_Integer; { Inc delta size }
- CONSTRUCTOR Init (ALimit, ADelta: Sw_Integer);
- CONSTRUCTOR Load (Var S: TStream);
- DESTRUCTOR Done; Virtual;
- FUNCTION At (Index: Sw_Integer): Pointer;
- FUNCTION IndexOf (Item: Pointer): Sw_Integer; Virtual;
- FUNCTION GetItem (Var S: TStream): Pointer; Virtual;
- FUNCTION LastThat (Test: Pointer): Pointer;
- FUNCTION FirstThat (Test: Pointer): Pointer;
- PROCEDURE Pack;
- PROCEDURE FreeAll;
- PROCEDURE DeleteAll;
- PROCEDURE Free (Item: Pointer);
- PROCEDURE Insert (Item: Pointer); Virtual;
- PROCEDURE Delete (Item: Pointer);
- PROCEDURE AtFree (Index: Sw_Integer);
- PROCEDURE FreeItem (Item: Pointer); Virtual;
- PROCEDURE AtDelete (Index: Sw_Integer);
- PROCEDURE ForEach (Action: Pointer);
- PROCEDURE SetLimit (ALimit: Sw_Integer); Virtual;
- PROCEDURE Error (Code, Info: Integer); Virtual;
- PROCEDURE AtPut (Index: Sw_Integer; Item: Pointer);
- PROCEDURE AtInsert (Index: Sw_Integer; Item: Pointer);
- PROCEDURE Store (Var S: TStream);
- PROCEDURE PutItem (Var S: TStream; Item: Pointer); Virtual;
- END;
- PCollection = ^TCollection;
- {---------------------------------------------------------------------------}
- { TSortedCollection OBJECT - SORTED COLLECTION ANCESTOR }
- {---------------------------------------------------------------------------}
- TYPE
- TSortedCollection = OBJECT (TCollection)
- Duplicates: Boolean; { Duplicates flag }
- CONSTRUCTOR Init (ALimit, ADelta: Sw_Integer);
- CONSTRUCTOR Load (Var S: TStream);
- FUNCTION KeyOf (Item: Pointer): Pointer; Virtual;
- FUNCTION IndexOf (Item: Pointer): Sw_Integer; Virtual;
- FUNCTION Compare (Key1, Key2: Pointer): Sw_Integer; Virtual;
- FUNCTION Search (Key: Pointer; Var Index: Sw_Integer): Boolean;Virtual;
- PROCEDURE Insert (Item: Pointer); Virtual;
- PROCEDURE Store (Var S: TStream);
- END;
- PSortedCollection = ^TSortedCollection;
- {---------------------------------------------------------------------------}
- { TStringCollection OBJECT - STRING COLLECTION OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- TStringCollection = OBJECT (TSortedCollection)
- FUNCTION GetItem (Var S: TStream): Pointer; Virtual;
- FUNCTION Compare (Key1, Key2: Pointer): Sw_Integer; Virtual;
- PROCEDURE FreeItem (Item: Pointer); Virtual;
- PROCEDURE PutItem (Var S: TStream; Item: Pointer); Virtual;
- END;
- PStringCollection = ^TStringCollection;
- {---------------------------------------------------------------------------}
- { TStrCollection OBJECT - STRING COLLECTION OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- TStrCollection = OBJECT (TSortedCollection)
- FUNCTION Compare (Key1, Key2: Pointer): Sw_Integer; Virtual;
- FUNCTION GetItem (Var S: TStream): Pointer; Virtual;
- PROCEDURE FreeItem (Item: Pointer); Virtual;
- PROCEDURE PutItem (Var S: TStream; Item: Pointer); Virtual;
- END;
- PStrCollection = ^TStrCollection;
- { ******************************* REMARK ****************************** }
- { This is a completely >> NEW << object which holds a collection of }
- { strings but does not alphabetically sort them. It is a very useful }
- { object for insert ordered list boxes! }
- { ****************************** END REMARK *** Leon de Boer, 15May96 * }
- {---------------------------------------------------------------------------}
- { TUnSortedStrCollection - UNSORTED STRING COLLECTION OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- TUnSortedStrCollection = OBJECT (TStringCollection)
- PROCEDURE Insert (Item: Pointer); Virtual;
- END;
- PUnSortedStrCollection = ^TUnSortedStrCollection;
- {---------------------------------------------------------------------------}
- { TResourceCollection OBJECT - RESOURCE COLLECTION OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- TResourceCollection = OBJECT (TStringCollection)
- FUNCTION KeyOf (Item: Pointer): Pointer; Virtual;
- FUNCTION GetItem (Var S: TStream): Pointer; Virtual;
- PROCEDURE FreeItem (Item: Pointer); Virtual;
- PROCEDURE PutItem (Var S: TStream; Item: Pointer); Virtual;
- END;
- PResourceCollection = ^TResourceCollection;
- {---------------------------------------------------------------------------}
- { TResourceFile OBJECT - RESOURCE FILE OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- TResourceFile = OBJECT (TObject)
- Stream : PStream; { File as a stream }
- Modified: Boolean; { Modified flag }
- CONSTRUCTOR Init (AStream: PStream);
- DESTRUCTOR Done; Virtual;
- FUNCTION Count: Sw_Integer;
- FUNCTION KeyAt (I: Sw_Integer): String;
- FUNCTION Get (Key: String): PObject;
- FUNCTION SwitchTo (AStream: PStream; Pack: Boolean): PStream;
- PROCEDURE Flush;
- PROCEDURE Delete (Key: String);
- PROCEDURE Put (Item: PObject; Key: String);
- PRIVATE
- BasePos: LongInt; { Base position }
- IndexPos: LongInt; { Index position }
- Index: TResourceCollection; { Index collection }
- END;
- PResourceFile = ^TResourceFile;
- TYPE
- TStrIndexRec = Packed RECORD
- Key, Count, Offset: Word;
- END;
- TStrIndex = Array [0..9999] Of TStrIndexRec;
- PStrIndex = ^TStrIndex;
- {---------------------------------------------------------------------------}
- { TStringList OBJECT - STRING LIST OBJECT }
- {---------------------------------------------------------------------------}
- TStringList = OBJECT (TObject)
- CONSTRUCTOR Load (Var S: TStream);
- DESTRUCTOR Done; Virtual;
- FUNCTION Get (Key: Sw_Word): String;
- PRIVATE
- Stream : PStream;
- BasePos : Longint;
- IndexSize: Sw_Word;
- Index : PStrIndex;
- PROCEDURE ReadStr (Var S: String; Offset, Skip: Sw_Word);
- END;
- PStringList = ^TStringList;
- {---------------------------------------------------------------------------}
- { TStrListMaker OBJECT - RESOURCE FILE OBJECT }
- {---------------------------------------------------------------------------}
- TYPE
- TStrListMaker = OBJECT (TObject)
- CONSTRUCTOR Init (AStrSize, AIndexSize: Sw_Word);
- DESTRUCTOR Done; Virtual;
- PROCEDURE Put (Key: Sw_Word; S: String);
- PROCEDURE Store (Var S: TStream);
- PRIVATE
- StrPos : Sw_Word;
- StrSize : Sw_Word;
- Strings : PByteArray;
- IndexPos : Sw_Word;
- IndexSize: Sw_Word;
- Index : PStrIndex;
- Cur : TStrIndexRec;
- PROCEDURE CloseCurrent;
- END;
- PStrListMaker = ^TStrListMaker;
- {***************************************************************************}
- { INTERFACE ROUTINES }
- {***************************************************************************}
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { DYNAMIC STRING INTERFACE ROUTINES }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {-NewStr-------------------------------------------------------------
- Allocates a dynamic string into memory. If S is nil, NewStr returns
- a nil pointer, otherwise NewStr allocates Length(S)+1 bytes of memory
- containing a copy of S, and returns a pointer to the string.
- 12Jun96 LdB
- ---------------------------------------------------------------------}
- FUNCTION NewStr (Const S: String): PString;
- {-DisposeStr---------------------------------------------------------
- Disposes of a PString allocated by the function NewStr.
- 12Jun96 LdB
- ---------------------------------------------------------------------}
- PROCEDURE DisposeStr (P: PString);
- PROCEDURE SetStr(VAR p:pString; CONST s:STRING);
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { STREAM INTERFACE ROUTINES }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {-Abstract-----------------------------------------------------------
- Terminates program with a run-time error 211. When implementing
- an abstract object type, call Abstract in those virtual methods that
- must be overridden in descendant types. This ensures that any
- attempt to use instances of the abstract object type will fail.
- 12Jun96 LdB
- ---------------------------------------------------------------------}
- PROCEDURE Abstract;
- {-RegisterObjects----------------------------------------------------
- Registers the three standard objects TCollection, TStringCollection
- and TStrCollection.
- 02Sep97 LdB
- ---------------------------------------------------------------------}
- PROCEDURE RegisterObjects;
- {-RegisterType-------------------------------------------------------
- Registers the given object type with Free Vision's streams, creating
- a list of known objects. Streams can only store and return these known
- object types. Each registered object needs a unique stream registration
- record, of type TStreamRec.
- 02Sep97 LdB
- ---------------------------------------------------------------------}
- PROCEDURE RegisterType (Var S: TStreamRec);
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { GENERAL FUNCTION INTERFACE ROUTINES }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {-LongMul------------------------------------------------------------
- Returns the long integer value of X * Y integer values.
- 04Sep97 LdB
- ---------------------------------------------------------------------}
- FUNCTION LongMul (X, Y: Integer): LongInt;
- {-LongDiv------------------------------------------------------------
- Returns the integer value of long integer X divided by integer Y.
- 04Sep97 LdB
- ---------------------------------------------------------------------}
- FUNCTION LongDiv (X: Longint; Y: Integer): Integer;
- {***************************************************************************}
- { PUBLIC INITIALIZED VARIABLES }
- {***************************************************************************}
- CONST
- {---------------------------------------------------------------------------}
- { INITIALIZED DOS/DPMI/WIN/OS2 PUBLIC VARIABLES }
- {---------------------------------------------------------------------------}
- StreamError: Pointer = Nil; { Stream error ptr }
- DefaultTPCompatible: Boolean = false;
- {---------------------------------------------------------------------------}
- { STREAM REGISTRATION RECORDS }
- {---------------------------------------------------------------------------}
- CONST
- RCollection: TStreamRec = (
- ObjType: 50;
- VmtLink: Ofs(TypeOf(TCollection)^);
- Load: @TCollection.Load;
- Store: @TCollection.Store;
- Next: Nil);
- RStringCollection: TStreamRec = (
- ObjType: 51;
- VmtLink: Ofs(TypeOf(TStringCollection)^);
- Load: @TStringCollection.Load;
- Store: @TStringCollection.Store;
- Next: Nil);
- RStrCollection: TStreamRec = (
- ObjType: 69;
- VmtLink: Ofs(TypeOf(TStrCollection)^);
- Load: @TStrCollection.Load;
- Store: @TStrCollection.Store;
- Next: Nil);
- RStringList: TStreamRec = (
- ObjType: 52;
- VmtLink: Ofs(TypeOf(TStringList)^);
- Load: @TStringList.Load;
- Store: Nil;
- Next: Nil);
- RStrListMaker: TStreamRec = (
- ObjType: 52;
- VmtLink: Ofs(TypeOf(TStrListMaker)^);
- Load: Nil;
- Store: @TStrListMaker.Store;
- Next: Nil);
- {<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>}
- IMPLEMENTATION
- {<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>}
- Uses dos;
- {***************************************************************************}
- { HELPER ROUTINES FOR CALLING }
- {***************************************************************************}
- type
- FramePointer = pointer;
- PointerLocal = function(_EBP: FramePointer; Param1: pointer): pointer;
- PointerConstructor = function(VMT: pointer; Obj: pointer; Param1: pointer): pointer;
- PointerMethod = function(Obj: pointer; Param1: pointer): pointer;
- function PreviousFramePointer: FramePointer;assembler;
- {$undef FPC_PreviousFramePointer_Implemented}
- asm
- {$ifdef i386}
- {$define FPC_PreviousFramePointer_Implemented}
- movl (%ebp), %eax
- end ['EAX'];
- {$endif}
- {$ifdef m68k}
- {$define FPC_PreviousFramePointer_Implemented}
- move.l (a6),d0
- end ['D0'];
- {$endif}
- {$ifdef powerpc}
- {$define FPC_PreviousFramePointer_Implemented}
- {$warning FIX ME !!!! }
- { getting the previous stack frame is quite hard for the standard powerpc calling conventions
- because we don't know the size of the locals, it seems that we need some compiler magic for this
- }
- end;
- {$endif}
- {$ifndef FPC_PreviousFramePointer_Implemented}
- {$error PreviousFramePointer function not implemented}
- {$endif not FPC_PreviousFramePointer_Implemented}
- function CallPointerConstructor(Ctor: pointer; Obj: pointer; VMT: pointer; Param1: pointer): pointer;
- {$undef FPC_CallPointerConstructor_Implemented}
- begin
- asm
- {$ifdef i386}
- {$define FPC_CallPointerConstructor_Implemented}
- movl Obj, %esi
- {$endif}
- {$ifdef m68k}
- {$define FPC_CallPointerConstructor_Implemented}
- move.l Obj, a5
- {$endif}
- end;
- CallPointerConstructor := PointerConstructor(Ctor)(VMT, Obj, Param1)
- end;
- {$ifdef powerpc}
- {$define FPC_CallPointerConstructor_Implemented}
- { for the powerpc, we don't need to load self, because we use standard calling conventions
- so self should be in a register anyways }
- {$endif}
- {$ifndef FPC_CallPointerConstructor_Implemented}
- {$error CallPointerConstructor function not implemented}
- {$endif not FPC_CallPointerConstructor_Implemented}
- function CallPointerMethod(Method: pointer; Obj: pointer; Param1: pointer): pointer;
- {$undef FPC_CallPointerMethod_Implemented}
- begin
- asm
- {$ifdef i386}
- {$define FPC_CallPointerMethod_Implemented}
- movl Obj, %esi
- {$endif}
- {$ifdef m68k}
- {$define FPC_CallPointerMethod_Implemented}
- move.l Obj, a5
- {$endif}
- {$ifdef powerpc}
- {$define FPC_CallPointerMethod_Implemented}
- { for the powerpc, we don't need to load self, because we use standard calling conventions
- so self should be in a register anyways }
- {$endif}
- end;
- CallPointerMethod := PointerMethod(Method)(Obj, Param1)
- end;
- {$ifndef FPC_CallPointerMethod_Implemented}
- {$error CallPointerMethod function not implemented}
- {$endif not FPC_CallPointerMethod_Implemented}
- function CallPointerLocal(Func: pointer; Frame: FramePointer; Param1: pointer): pointer;
- begin
- CallPointerLocal := PointerLocal(Func)(Frame, Param1)
- end;
- {***************************************************************************}
- { PRIVATE INITIALIZED VARIABLES }
- {***************************************************************************}
- {---------------------------------------------------------------------------}
- { INITIALIZED DOS/DPMI/WIN/OS2 PRIVATE VARIABLES }
- {---------------------------------------------------------------------------}
- CONST
- StreamTypes: PStreamRec = Nil; { Stream types reg }
- {***************************************************************************}
- { PRIVATE INTERNAL ROUTINES }
- {***************************************************************************}
- {---------------------------------------------------------------------------}
- { RegisterError -> Platforms DOS/DPMI/WINDOWS/OS2 - Checked 12Jun96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE RegisterError;
- BEGIN
- RunError(212); { Register error }
- END;
- {***************************************************************************}
- { OBJECT METHODS }
- {***************************************************************************}
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TRect OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- PROCEDURE CheckEmpty (Var Rect: TRect);
- BEGIN
- With Rect Do Begin
- If (A.X >= B.X) OR (A.Y >= B.Y) Then Begin { Zero or reversed }
- A.X := 0; { Clear a.x }
- A.Y := 0; { Clear a.y }
- B.X := 0; { Clear b.x }
- B.Y := 0; { Clear b.y }
- End;
- End;
- END;
- {--TRect--------------------------------------------------------------------}
- { Empty -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TRect.Empty: Boolean;
- BEGIN
- Empty := (A.X >= B.X) OR (A.Y >= B.Y); { Empty result }
- END;
- {--TRect--------------------------------------------------------------------}
- { Equals -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TRect.Equals (R: TRect): Boolean;
- BEGIN
- Equals := (A.X = R.A.X) AND (A.Y = R.A.Y) AND
- (B.X = R.B.X) AND (B.Y = R.B.Y); { Equals result }
- END;
- {--TRect--------------------------------------------------------------------}
- { Contains -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TRect.Contains (P: TPoint): Boolean;
- BEGIN
- Contains := (P.X >= A.X) AND (P.X < B.X) AND
- (P.Y >= A.Y) AND (P.Y < B.Y); { Contains result }
- END;
- {--TRect--------------------------------------------------------------------}
- { Copy -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TRect.Copy (R: TRect);
- BEGIN
- A := R.A; { Copy point a }
- B := R.B; { Copy point b }
- END;
- {--TRect--------------------------------------------------------------------}
- { Union -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TRect.Union (R: TRect);
- BEGIN
- If (R.A.X < A.X) Then A.X := R.A.X; { Take if smaller }
- If (R.A.Y < A.Y) Then A.Y := R.A.Y; { Take if smaller }
- If (R.B.X > B.X) Then B.X := R.B.X; { Take if larger }
- If (R.B.Y > B.Y) Then B.Y := R.B.Y; { Take if larger }
- END;
- {--TRect--------------------------------------------------------------------}
- { Intersect -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TRect.Intersect (R: TRect);
- BEGIN
- If (R.A.X > A.X) Then A.X := R.A.X; { Take if larger }
- If (R.A.Y > A.Y) Then A.Y := R.A.Y; { Take if larger }
- If (R.B.X < B.X) Then B.X := R.B.X; { Take if smaller }
- If (R.B.Y < B.Y) Then B.Y := R.B.Y; { Take if smaller }
- CheckEmpty(Self); { Check if empty }
- END;
- {--TRect--------------------------------------------------------------------}
- { Move -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TRect.Move (ADX, ADY: Sw_Integer);
- BEGIN
- Inc(A.X, ADX); { Adjust A.X }
- Inc(A.Y, ADY); { Adjust A.Y }
- Inc(B.X, ADX); { Adjust B.X }
- Inc(B.Y, ADY); { Adjust B.Y }
- END;
- {--TRect--------------------------------------------------------------------}
- { Grow -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TRect.Grow (ADX, ADY: Sw_Integer);
- BEGIN
- Dec(A.X, ADX); { Adjust A.X }
- Dec(A.Y, ADY); { Adjust A.Y }
- Inc(B.X, ADX); { Adjust B.X }
- Inc(B.Y, ADY); { Adjust B.Y }
- CheckEmpty(Self); { Check if empty }
- END;
- {--TRect--------------------------------------------------------------------}
- { Assign -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TRect.Assign (XA, YA, XB, YB: Sw_Integer);
- BEGIN
- A.X := XA; { Hold A.X value }
- A.Y := YA; { Hold A.Y value }
- B.X := XB; { Hold B.X value }
- B.Y := YB; { Hold B.Y value }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TObject OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- TYPE
- DummyObject = OBJECT (TObject) { Internal object }
- Data: RECORD END; { Helps size VMT link }
- END;
- { ******************************* REMARK ****************************** }
- { I Prefer this code because it self sizes VMT link rather than using a }
- { fixed record structure thus it should work on all compilers without a }
- { specific record to match each compiler. }
- { ****************************** END REMARK *** Leon de Boer, 10May96 * }
- {--TObject------------------------------------------------------------------}
- { Init -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TObject.Init;
- VAR LinkSize: LongInt; Dummy: DummyObject;
- BEGIN
- LinkSize := LongInt(@Dummy.Data)-LongInt(@Dummy); { Calc VMT link size }
- FillChar(Pointer(LongInt(@Self)+LinkSize)^,
- SizeOf(Self)-LinkSize, #0); { Clear data fields }
- END;
- {--TObject------------------------------------------------------------------}
- { Free -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TObject.Free;
- BEGIN
- Dispose(PObject(@Self), Done); { Dispose of self }
- END;
- {--TObject------------------------------------------------------------------}
- { Is_Object -> Platforms DOS/DPMI/WIN/OS2 - Checked 5Mar00 DM }
- {---------------------------------------------------------------------------}
- FUNCTION TObject.Is_Object(P:Pointer):Boolean;
- TYPE
- PVMT=^VMT;
- VMT=RECORD
- Size,NegSize:Longint;
- ParentLink:PVMT;
- END;
- VAR SP:^PVMT; Q:PVMT;
- BEGIN
- SP:=@SELF;
- Q:=SP^;
- Is_Object:=False;
- While Q<>Nil Do Begin
- IF Q=P THEN Begin
- Is_Object:=True;
- Break;
- End;
- Q:=Q^.Parentlink;
- End;
- END;
- {--TObject------------------------------------------------------------------}
- { Done -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- DESTRUCTOR TObject.Done;
- BEGIN { Abstract method }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TStream OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- CONSTRUCTOR TStream.Init;
- BEGIN
- Status := StOK;
- ErrorInfo := 0;
- StreamSize := 0;
- Position := 0;
- TPCompatible := DefaultTPCompatible;
- END;
- {--TStream------------------------------------------------------------------}
- { Get -> Platforms DOS/DPMI/WIN/OS2 - Checked 02Sep97 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TStream.Get: PObject;
- VAR ObjType: Sw_Word; P: PStreamRec; ObjTypeWord: Word;
- BEGIN
- If TPCompatible Then Begin
- { Read 16-bit word for TP compatibility. }
- Read(ObjTypeWord, SizeOf(ObjTypeWord));
- ObjType := ObjTypeWord
- End
- else
- Read(ObjType, SizeOf(ObjType)); { Read object type }
- If (ObjType<>0) Then Begin { Object registered }
- P := StreamTypes; { Current reg list }
- While (P <> Nil) AND (P^.ObjType <> ObjType) { Find object type OR }
- Do P := P^.Next; { Find end of chain }
- If (P=Nil) Then Begin { Not registered }
- Error(stGetError, ObjType); { Obj not registered }
- Get := Nil; { Return nil pointer }
- End Else
- Get :=PObject(
- CallPointerConstructor(P^.Load,Nil,P^.VMTLink, @Self)) { Call constructor }
- End Else Get := Nil; { Return nil pointer }
- END;
- {--TStream------------------------------------------------------------------}
- { StrRead -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TStream.StrRead: PChar;
- VAR L: Word; P: PChar;
- BEGIN
- Read(L, SizeOf(L)); { Read length }
- If (L = 0) Then StrRead := Nil Else Begin { Check for empty }
- GetMem(P, L + 1); { Allocate memory }
- If (P <> Nil) Then Begin { Check allocate okay }
- Read(P[0], L); { Read the data }
- P[L] := #0; { Terminate with #0 }
- End;
- StrRead := P; { Return PChar }
- End;
- END;
- {--TStream------------------------------------------------------------------}
- { ReadStr -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TStream.ReadStr: PString;
- VAR L: Byte; P: PString;
- BEGIN
- Read(L, 1); { Read string length }
- If (L > 0) Then Begin
- GetMem(P, L + 1); { Allocate memory }
- If (P <> Nil) Then Begin { Check allocate okay }
- P^[0] := Char(L); { Hold length }
- Read(P^[1], L); { Read string data }
- End;
- ReadStr := P; { Return string ptr }
- End Else ReadStr := Nil;
- END;
- {--TStream------------------------------------------------------------------}
- { GetPos -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TStream.GetPos: LongInt;
- BEGIN
- If (Status=stOk) Then GetPos := Position { Return position }
- Else GetPos := -1; { Stream in error }
- END;
- {--TStream------------------------------------------------------------------}
- { GetSize -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TStream.GetSize: LongInt;
- BEGIN
- If (Status=stOk) Then GetSize := StreamSize { Return stream size }
- Else GetSize := -1; { Stream in error }
- END;
- {--TStream------------------------------------------------------------------}
- { Close -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.Close;
- BEGIN { Abstract method }
- END;
- {--TStream------------------------------------------------------------------}
- { Reset -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.Reset;
- BEGIN
- Status := stOK; { Clear status }
- ErrorInfo := 0; { Clear error info }
- END;
- {--TStream------------------------------------------------------------------}
- { Flush -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.Flush;
- BEGIN { Abstract method }
- END;
- {--TStream------------------------------------------------------------------}
- { Truncate -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.Truncate;
- BEGIN
- Abstract; { Abstract error }
- END;
- {--TStream------------------------------------------------------------------}
- { Put -> Platforms DOS/DPMI/WIN/OS2 - Checked 02Sep97 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.Put (P: PObject);
- VAR ObjType: Sw_Word; Link: pointer; Q: PStreamRec; VmtPtr: ^pointer;
- ObjTypeWord: Word;
- BEGIN
- VmtPtr := Pointer(P); { Xfer object to ptr }
- if assigned(vmtptr) then
- Link := VmtPtr^ { VMT link }
- else
- Link:=nil;
- ObjType := 0; { Set objtype to zero }
- If (P<>Nil) AND (Link<>Nil) Then Begin { We have a VMT link }
- Q := StreamTypes; { Current reg list }
- While (Q <> Nil) AND (Q^.VMTLink <> Link) { Find link match OR }
- Do Q := Q^.Next; { Find end of chain }
- If (Q=Nil) Then Begin { End of chain found }
- Error(stPutError, 0); { Not registered error }
- Exit; { Now exit }
- End Else ObjType := Q^.ObjType; { Update object type }
- End;
- If TPCompatible Then Begin
- ObjTypeWord := word(ObjType);
- Write(ObjTypeWord, SizeOf(ObjTypeWord))
- end
- else
- Write(ObjType, SizeOf(ObjType)); { Write object type }
- If (ObjType<>0) Then { Registered object }
- CallPointerMethod(Q^.Store, P, @Self);
- END;
- {--TStream------------------------------------------------------------------}
- { Seek -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.Seek (Pos: LongInt);
- BEGIN
- If (Status = stOk) Then Begin { Check status }
- If (Pos < 0) Then Pos := 0; { Remove negatives }
- If (Pos <= StreamSize) Then Position := Pos { If valid set pos }
- Else Error(stSeekError, Pos); { Position error }
- End;
- END;
- {--TStream------------------------------------------------------------------}
- { StrWrite -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.StrWrite (P: PChar);
- VAR L: Word; Q: PByteArray;
- BEGIN
- L := 0; { Preset zero size }
- Q := PByteArray(P); { Transfer type }
- If (Q <> Nil) Then While (Q^[L] <> 0) Do Inc(L); { PChar length }
- Write(L, SizeOf(L)); { Store length }
- If (P <> Nil) Then Write(P[0], L); { Write data }
- END;
- {--TStream------------------------------------------------------------------}
- { WriteStr -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.WriteStr (P: PString);
- CONST Empty: String[1] = '';
- BEGIN
- If (P <> Nil) Then Write(P^, Length(P^) + 1) { Write string }
- Else Write(Empty, 1); { Write empty string }
- END;
- {--TStream------------------------------------------------------------------}
- { Open -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.Open (OpenMode: Word);
- BEGIN { Abstract method }
- END;
- {--TStream------------------------------------------------------------------}
- { Error -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.Error (Code, Info: Integer);
- TYPE TErrorProc = Procedure(Var S: TStream);
- BEGIN
- Status := Code; { Hold error code }
- ErrorInfo := Info; { Hold error info }
- If (StreamError <> Nil) Then
- TErrorProc(StreamError)(Self); { Call error ptr }
- END;
- {--TStream------------------------------------------------------------------}
- { Read -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.Read (Var Buf; Count: Sw_Word);
- BEGIN
- Abstract; { Abstract error }
- END;
- {--TStream------------------------------------------------------------------}
- { Write -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.Write (Var Buf; Count: Sw_Word);
- BEGIN
- Abstract; { Abstract error }
- END;
- {--TStream------------------------------------------------------------------}
- { CopyFrom -> Platforms DOS/DPMI/WIN/OS2 - Checked 10May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStream.CopyFrom (Var S: TStream; Count: Longint);
- VAR W: Word; Buffer: Array[0..1023] of Byte;
- BEGIN
- While (Count > 0) Do Begin
- If (Count > SizeOf(Buffer)) Then { To much data }
- W := SizeOf(Buffer) Else W := Count; { Size to transfer }
- S.Read(Buffer, W); { Read from stream }
- Write(Buffer, W); { Write to stream }
- Dec(Count, W); { Dec write count }
- End;
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TDosStream OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {$IFOPT I+}
- {$DEFINE IO_CHECK_ON}
- {$I-}
- {$ENDIF}
- {--TDosStream---------------------------------------------------------------}
- { Init -> Platforms DOS/DPMI/WIN/OS2 - Checked 16May96 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TDosStream.Init (FileName: FNameStr; Mode: Word);
- VAR OldFileMode : Byte;
- DosStreamError : Word;
- BEGIN
- Inherited Init; { Call ancestor }
- FileName := FileName+#0; { Make asciiz }
- Move(FileName[1], FName, Length(FileName)); { Create asciiz name }
- Handle := InvalidHandle;
- Assign(FileInfo,FileName);
- { Handle the mode }
- if Mode =stCreate then
- Begin
- Rewrite(FileInfo,1);
- end
- else
- Begin
- OldFileMode := FileMode;
- FileMode := Mode and 3;
- System.Reset(FileInfo,1);
- FileMode := OldFileMode;
- { To use the correct mode we must reclose the file
- and open it again
- }
- end;
- Handle := FileRec(FileInfo).Handle; { Set handle value }
- DosStreamError := IOResult;
- If DosStreamError = 0 then
- Begin
- StreamSize := System.FileSize(FileInfo);
- end;
- If DosStreamError = 0 then
- DosStreamError := IOResult;
- If (DosStreamError <> 0) Then
- Error(stInitError, DosStreamError) { Call stream error }
- else
- Status := StOK;
- END;
- {--TDosStream---------------------------------------------------------------}
- { Done -> Platforms DOS/DPMI/WIN/OS2 - Checked 16May96 LdB }
- {---------------------------------------------------------------------------}
- DESTRUCTOR TDosStream.Done;
- var
- DosStreamError : Word;
- BEGIN
- if Handle <> InvalidHandle then
- Begin
- System.Close(FileInfo);
- DosStreamError := IOResult;
- If DosStreamError = 0 then
- Status := stOk
- else
- Error(stError, DosStreamError);
- end;
- Position := 0; { Zero the position }
- Handle := InvalidHandle;
- Inherited Done; { Call ancestor }
- END;
- {--TDosStream---------------------------------------------------------------}
- { Close -> Platforms DOS/DPMI/WIN/OS2 - Checked 16May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TDosStream.Close;
- var
- DosStreamError : Word;
- BEGIN
- if Handle <> InvalidHandle then { Is file closed ? }
- Begin
- System.Close(FileInfo); { Close file }
- DosStreamError := IOResult; { Check for error }
- If DosStreamError = 0 then
- Status := stOk
- else
- Error(stError, DosStreamError); { Call error routine }
- end;
- Position := 0; { Zero the position }
- Handle := InvalidHandle; { Handle invalid }
- END;
- {--TDosStream---------------------------------------------------------------}
- { Truncate -> Platforms DOS/DPMI/WIN/OS2 - Checked 16May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TDosStream.Truncate;
- var
- DosStreamError : Word;
- BEGIN
- If Status = stOk then
- Begin
- System.Truncate(FileInfo);
- DosStreamError := IOResult;
- If DosStreamError = 0 then
- { Status is already = stOK }
- StreamSize := Position
- else
- Error(stError, DosStreamError);
- end;
- END;
- {--TDosStream---------------------------------------------------------------}
- { Seek -> Platforms DOS/DPMI/WIN/OS2 - Checked 16May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TDosStream.Seek (Pos: Longint);
- var
- DosStreamError : Word;
- BEGIN
- If (Status=stOk) Then
- Begin { Check status okay }
- If (Pos < 0) Then
- Pos := 0; { Negatives removed }
- System.Seek(FileInfo, Pos);
- DosStreamError := IOResult;
- if DosStreamError <> 0 then
- Error(stSeekError, DosStreamError){ Specific seek error }
- Else Position := Pos; { Adjust position }
- { Status is already = stOK }
- End;
- END;
- {--TDosStream---------------------------------------------------------------}
- { Open -> Platforms DOS/DPMI/WIN/OS2 - Checked 16May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TDosStream.Open (OpenMode: Word);
- VAR OldFileMode : Byte;
- DosStreamError : Word;
- BEGIN
- If (Status=stOk) Then
- Begin { Check status okay }
- If (Handle = InvalidHandle) Then
- Begin { File not open }
- Assign(FileInfo,FName);
- { Handle the mode }
- if OpenMode =stCreate then
- Begin
- System.Rewrite(FileInfo,1);
- end
- else
- Begin
- OldFileMode := FileMode;
- FileMode := OpenMode and 3;
- System.Reset(FileInfo,1);
- FileMode := OldFileMode;
- { To use the correct mode we must reclose the file
- and open it again
- }
- end;
- Handle := FileRec(FileInfo).Handle; { Set handle value }
- DosStreamError := IOResult;
- If DosStreamError = 0 then
- StreamSize := System.FileSize(FileInfo);
- If DosStreamError = 0 then
- DosStreamError := IOResult;
- If (DosStreamError <> 0) Then
- Error(stOpenError, DosStreamError) { Call stream error }
- else
- Status := StOK;
- Position := 0;
- end
- Else
- Error(stOpenError, 104); { File already open }
- End;
- END;
- {--TDosStream---------------------------------------------------------------}
- { Read -> Platforms DOS/DPMI/WIN/OS2 - Checked 16May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TDosStream.Read (Var Buf; Count: Sw_Word);
- VAR BytesMoved: Sw_Word;
- DosStreamError : Word;
- BEGIN
- If Status = StOK then
- Begin
- If (Position + Count > StreamSize) Then { Insufficient data }
- Error(stReadError, 0); { Read beyond end!!! }
- If (Handle = InvalidHandle) Then
- Error(stReadError, 103); { File not open }
- BlockRead(FileInfo, Buf, Count, BytesMoved); { Read from file }
- DosStreamError := IOResult;
- If ((DosStreamError<>0) OR (BytesMoved<>Count)) Then
- Begin { Error was detected }
- BytesMoved := 0; { Clear bytes moved }
- If (DosStreamError <> 0) Then
- Error(stReadError, DosStreamError) { Specific read error }
- Else
- Error(stReadError, 0); { Non specific error }
- End;
- Inc(Position, BytesMoved); { Adjust position }
- End;
- { If there was already an error, or an error was just
- generated, fill the vuffer with NULL
- }
- If Status <> StOK then
- FillChar(Buf, Count, #0); { Error clear buffer }
- END;
- {--TDosStream---------------------------------------------------------------}
- { Write -> Platforms DOS/DPMI/WIN/OS2 - Checked 16May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TDosStream.Write (Var Buf; Count: Sw_Word);
- VAR BytesMoved: Sw_Word;
- DosStreamError : Word;
- BEGIN
- { If status is not OK, simply exit }
- if Status <> StOK then
- exit;
- If (Handle = InvalidHandle) Then
- Error(stWriteError, 103); { File not open }
- BlockWrite(FileInfo, Buf, Count, BytesMoved); { Write to file }
- DosStreamError := IOResult;
- If ((DosStreamError<>0) OR (BytesMoved<>Count)) Then
- Begin { Error was detected }
- BytesMoved := 0; { Clear bytes moved }
- If (DosStreamError<>0) Then
- Error(stWriteError, DOSStreamError) { Specific write error }
- Else
- Error(stWriteError, 0); { Non specific error }
- End;
- Inc(Position, BytesMoved); { Adjust position }
- If (Position > StreamSize) Then { File expanded }
- StreamSize := Position; { Adjust stream size }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TBufStream OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TBufStream---------------------------------------------------------------}
- { Init -> Platforms DOS/DPMI/WIN/OS2 - Checked 17May96 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TBufStream.Init (FileName: FNameStr; Mode, Size: Word);
- BEGIN
- Inherited Init(FileName, Mode); { Call ancestor }
- BufSize := Size; { Hold buffer size }
- If (Size<>0) Then GetMem(Buffer, Size); { Allocate buffer }
- If (Buffer=Nil) Then Error(stInitError, 0); { Buffer allocate fail }
- END;
- {--TBufStream---------------------------------------------------------------}
- { Done -> Platforms DOS/DPMI/WIN/OS2 - Checked 17May96 LdB }
- {---------------------------------------------------------------------------}
- DESTRUCTOR TBufStream.Done;
- BEGIN
- Flush; { Flush the file }
- Inherited Done; { Call ancestor }
- If (Buffer<>Nil) Then FreeMem(Buffer, BufSize); { Release buffer }
- END;
- {--TBufStream---------------------------------------------------------------}
- { Close -> Platforms DOS/DPMI/WIN/OS2 - Checked 17May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TBufStream.Close;
- BEGIN
- Flush; { Flush the buffer }
- Inherited Close; { Call ancestor }
- END;
- {--TBufStream---------------------------------------------------------------}
- { Flush -> Platforms DOS/DPMI/WIN/OS2 - Checked 17May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TBufStream.Flush;
- VAR W: Sw_Word;
- DosStreamError : Word;
- BEGIN
- If Status <> StOK then
- exit;
- If (LastMode=2) AND (BufPtr<>0) Then Begin { Must update file }
- If (Handle = InvalidHandle) Then DosStreamError := 103 { File is not open }
- Else
- Begin
- BlockWrite(FileInfo, Buffer^,BufPtr, W); { Write to file }
- DosStreamError := IOResult;
- End;
- If (DosStreamError<>0) OR (W<>BufPtr) Then { We have an error }
- If (DosStreamError=0) Then Error(stWriteError, 0){ Unknown write error }
- Else Error(stError, DosStreamError); { Specific write error }
- End;
- BufPtr := 0; { Reset buffer ptr }
- BufEnd := 0; { Reset buffer end }
- END;
- {--TBufStream---------------------------------------------------------------}
- { Truncate -> Platforms DOS/DPMI/WIN/OS2 - Checked 17May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TBufStream.Truncate;
- BEGIN
- Flush; { Flush buffer }
- Inherited Truncate; { Truncate file }
- END;
- {--TBufStream---------------------------------------------------------------}
- { Seek -> Platforms DOS/DPMI/WIN/OS2 - Checked 17May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TBufStream.Seek (Pos: LongInt);
- BEGIN
- If (Status=stOk) Then Begin { Check status okay }
- If (Position<>Pos) Then Begin { Move required }
- Flush; { Flush the buffer }
- Inherited Seek(Pos); { Call ancestor }
- End;
- End;
- END;
- {--TBufStream---------------------------------------------------------------}
- { Open -> Platforms DOS/DPMI/WIN/OS2 - Checked 17May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TBufStream.Open (OpenMode: Word);
- BEGIN
- If (Status=stOk) Then Begin { Check status okay }
- BufPtr := 0; { Clear buffer start }
- BufEnd := 0; { Clear buffer end }
- Inherited Open(OpenMode); { Call ancestor }
- End;
- END;
- {--TBufStream---------------------------------------------------------------}
- { Read -> Platforms DOS/DPMI/WIN/OS2 - Checked 17May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TBufStream.Read (Var Buf; Count: Sw_Word);
- VAR Success: Integer; W, Bw: Sw_Word; P: PByteArray;
- DosStreamError : Word;
- BEGIN
- If Status <> StOk then
- begin
- FillChar(Buf, Count, #0); { Error clear buffer }
- exit;
- end;
- If (Position + Count > StreamSize) Then { Read pas stream end }
- Error(stReadError, 0); { Call stream error }
- If (Handle = InvalidHandle) Then Error(stReadError, 103); { File not open }
- P := @Buf; { Transfer address }
- If (LastMode=2) Then Flush; { Flush write buffer }
- LastMode := 1; { Now set read mode }
- While (Count>0) AND (Status=stOk) Do Begin { Check status & count }
- If (BufPtr=BufEnd) Then Begin { Buffer is empty }
- If (Position + BufSize > StreamSize) Then
- Bw := StreamSize - Position { Amount of file left }
- Else Bw := BufSize; { Full buffer size }
- BlockRead(FileInfo, Buffer^, Bw, W);
- DosStreamError := IOResult; { Read from file }
- If ((DosStreamError<>0) OR (Bw<>W)) Then Begin { Error was detected }
- If (DosStreamError<>0) Then
- Error(stReadError, DosStreamError) { Specific read error }
- Else Error(stReadError, 0); { Non specific error }
- End Else Begin
- BufPtr := 0; { Reset BufPtr }
- BufEnd := W; { End of buffer }
- End;
- End;
- If (Status=stOk) Then Begin { Status still okay }
- W := BufEnd - BufPtr; { Space in buffer }
- If (Count < W) Then W := Count; { Set transfer size }
- Move(Buffer^[BufPtr], P^, W); { Data from buffer }
- Dec(Count, W); { Reduce count }
- Inc(BufPtr, W); { Advance buffer ptr }
- P := Pointer(LongInt(P) + W); { Transfer address }
- Inc(Position, W); { Advance position }
- End;
- End;
- If (Status<>stOk) AND (Count>0) Then
- FillChar(P^, Count, #0); { Error clear buffer }
- END;
- {--TBufStream---------------------------------------------------------------}
- { Write -> Platforms DOS/DPMI/WIN/OS2 - Checked 17May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TBufStream.Write (Var Buf; Count: Sw_Word);
- VAR Success: Integer; W: Sw_Word; P: PByteArray;
- DosStreamError : Word;
- BEGIN
- if Status <> StOK then exit; { Exit if error }
- If (Handle = InvalidHandle) Then Error(stWriteError, 103); { File not open }
- If (LastMode=1) Then Flush; { Flush read buffer }
- LastMode := 2; { Now set write mode }
- P := @Buf; { Transfer address }
- While (Count>0) AND (Status=stOk) Do Begin { Check status & count }
- If (BufPtr=BufSize) Then Begin { Buffer is full }
- BlockWrite(FileInfo, Buffer^, BufSize,W); { Write to file }
- DosStreamError := IOResult;
- If (DosStreamError<>0) OR (W<>BufSize) Then { We have an error }
- If (DosStreamError=0) Then Error(stWriteError, 0) { Unknown write error }
- Else Error(stError, DosStreamError); { Specific write error }
- BufPtr := 0; { Reset BufPtr }
- End;
- If (Status=stOk) Then Begin { Status still okay }
- W := BufSize - BufPtr; { Space in buffer }
- If (Count < W) Then W := Count; { Transfer size }
- Move(P^, Buffer^[BufPtr], W); { Data to buffer }
- Dec(Count, W); { Reduce count }
- Inc(BufPtr, W); { Advance buffer ptr }
- P := Pointer(LongInt(P) + W); { Transfer address }
- Inc(Position, W); { Advance position }
- If (Position > StreamSize) Then { File has expanded }
- StreamSize := Position; { Update new size }
- End;
- End;
- END;
- {$IFDEF IO_CHECK_ON}
- {$UNDEF IO_CHECK_ON}
- {$I+}
- {$ENDIF}
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TMemoryStream OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TMemoryStream------------------------------------------------------------}
- { Init -> Platforms DOS/DPMI/WIN/OS2 - Checked 19May96 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TMemoryStream.Init (ALimit: LongInt; ABlockSize: Word);
- VAR W: Word;
- BEGIN
- Inherited Init; { Call ancestor }
- If (ABlockSize=0) Then BlkSize := 8192 Else { Default blocksize }
- BlkSize := ABlockSize; { Set blocksize }
- If (ALimit = 0) Then W := 1 Else { At least 1 block }
- W := (ALimit + BlkSize - 1) DIV BlkSize; { Blocks needed }
- If NOT ChangeListSize(W) Then { Try allocate blocks }
- Error(stInitError, 0); { Initialize error }
- END;
- {--TMemoryStream------------------------------------------------------------}
- { Done -> Platforms DOS/DPMI/WIN/OS2 - Checked 19May96 LdB }
- {---------------------------------------------------------------------------}
- DESTRUCTOR TMemoryStream.Done;
- BEGIN
- ChangeListSize(0); { Release all memory }
- Inherited Done; { Call ancestor }
- END;
- {--TMemoryStream------------------------------------------------------------}
- { Truncate -> Platforms DOS/DPMI/WIN/OS2 - Checked 19May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TMemoryStream.Truncate;
- VAR W: Word;
- BEGIN
- If (Status=stOk) Then Begin { Check status okay }
- If (Position = 0) Then W := 1 Else { At least one block }
- W := (Position + BlkSize - 1) DIV BlkSize; { Blocks needed }
- If ChangeListSize(W) Then StreamSize := Position { Set stream size }
- Else Error(stError, 0); { Error truncating }
- End;
- END;
- {--TMemoryStream------------------------------------------------------------}
- { Read -> Platforms DOS/DPMI/WIN/OS2 - Checked 19May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TMemoryStream.Read (Var Buf; Count: Sw_Word);
- VAR W, CurBlock, BlockPos: Word; Li: LongInt; P, Q: PByteArray;
- BEGIN
- If (Position + Count > StreamSize) Then { Insufficient data }
- Error(stReadError, 0); { Read beyond end!!! }
- P := @Buf; { Transfer address }
- While (Count>0) AND (Status=stOk) Do Begin { Check status & count }
- CurBlock := Position DIV BlkSize; { Current block }
- { * REMARK * - Do not shorten this, result can be > 64K }
- Li := CurBlock; { Transfer current block }
- Li := Li * BlkSize; { Current position }
- { * REMARK END * - Leon de Boer }
- BlockPos := Position - Li; { Current position }
- W := BlkSize - BlockPos; { Current block space }
- If (W > Count) Then W := Count; { Adjust read size }
- Q := Pointer(LongInt(BlkList^[CurBlock]) +
- BlockPos); { Calc pointer }
- Move(Q^, P^, W); { Move data to buffer }
- Inc(Position, W); { Adjust position }
- P := Pointer(LongInt(P) + W); { Transfer address }
- Dec(Count, W); { Adjust count left }
- End;
- If (Count<>0) Then FillChar(P^, Count, #0); { Error clear buffer }
- END;
- {--TMemoryStream------------------------------------------------------------}
- { Write -> Platforms DOS/DPMI/WIN/OS2 - Checked 19May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TMemoryStream.Write (Var Buf; Count: Sw_Word);
- VAR W, CurBlock, BlockPos: Word; Li: LongInt; P, Q: PByteArray;
- BEGIN
- If (Position + Count > MemSize) Then Begin { Expansion needed }
- If (Position + Count = 0) Then W := 1 Else { At least 1 block }
- W := (Position+Count+BlkSize-1) DIV BlkSize; { Blocks needed }
- If NOT ChangeListSize(W) Then
- Error(stWriteError, 0); { Expansion failed!!! }
- End;
- P := @Buf; { Transfer address }
- While (Count>0) AND (Status=stOk) Do Begin { Check status & count }
- CurBlock := Position DIV BlkSize; { Current segment }
- { * REMARK * - Do not shorten this, result can be > 64K }
- Li := CurBlock; { Transfer current block }
- Li := Li * BlkSize; { Current position }
- { * REMARK END * - Leon de Boer }
- BlockPos := Position - Li; { Current position }
- W := BlkSize - BlockPos; { Current block space }
- If (W > Count) Then W := Count; { Adjust write size }
- Q := Pointer(LongInt(BlkList^[CurBlock]) +
- BlockPos); { Calc pointer }
- Move(P^, Q^, W); { Transfer data }
- Inc(Position, W); { Adjust position }
- P := Pointer(LongInt(P) + W); { Transfer address }
- Dec(Count, W); { Adjust count left }
- If (Position > StreamSize) Then { File expanded }
- StreamSize := Position; { Adjust stream size }
- End;
- END;
- {***************************************************************************}
- { TMemoryStream PRIVATE METHODS }
- {***************************************************************************}
- {--TMemoryStream------------------------------------------------------------}
- { ChangeListSize -> Platforms DOS/DPMI/WIN/OS2 - Checked 19May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TMemoryStream.ChangeListSize (ALimit: Sw_Word): Boolean;
- VAR I, W: Word; Li: LongInt; P: PPointerArray;
- OldVal : Boolean;
- BEGIN
- If (ALimit <> BlkCount) Then Begin { Change is needed }
- ChangeListSize := False; { Preset failure }
- If (ALimit > MaxPtrs) Then Exit; { To many blocks req }
- If (ALimit <> 0) Then Begin { Create segment list }
- Li := ALimit * SizeOf(Pointer); { Block array size }
- If (MaxAvail > Li) Then Begin
- GetMem(P, Li); { Allocate memory }
- FillChar(P^, Li, #0); { Clear the memory }
- End Else Begin
- OldVal:=ReturnNilIfGrowHeapFails;
- ReturnNilIfGrowHeapFails:=true;
- GetMem(P,Li);
- ReturnNilIfGrowHeapFails:=OldVal;
- If P = Nil Then Exit;
- FillChar(P^, Li, #0); { Clear the memory }
- End; { Insufficient memory }
- If (BlkCount <> 0) AND (BlkList <> Nil) Then { Current list valid }
- If (BlkCount <= ALimit) Then Move(BlkList^,
- P^, BlkCount * SizeOf(Pointer)) Else { Move whole old list }
- Move(BlkList^, P^, Li); { Move partial list }
- End Else P := Nil; { No new block list }
- If (ALimit < BlkCount) Then { Shrink stream size }
- For W := BlkCount-1 DownTo ALimit Do
- FreeMem(BlkList^[W], BlkSize); { Release memory block }
- If (P <> Nil) AND (ALimit > BlkCount) Then Begin { Expand stream size }
- For W := BlkCount To ALimit-1 Do Begin
- If (MaxAvail < BlkSize) Then Begin { Check enough memory }
- OldVal:=ReturnNilIfGrowHeapFails;
- ReturnNilIfGrowHeapFails:=true;
- GetMem(P^[W],BlkSize);
- ReturnNilIfGrowHeapFails:=OldVal;
- If P = Nil Then Begin
- For I := BlkCount To W-1 Do
- FreeMem(P^[I], BlkSize); { Free mem allocated }
- FreeMem(P, Li); { Release memory }
- Exit;
- End { Now exit }
- End Else GetMem(P^[W], BlkSize); { Allocate memory }
- End;
- End;
- If (BlkCount <> 0) AND (BlkList<>Nil) Then
- FreeMem(BlkList, BlkCount * SizeOf(Pointer)); { Release old list }
- BlkList := P; { Hold new block list }
- BlkCount := ALimit; { Hold new count }
- { * REMARK * - Do not shorten this, result can be > 64K }
- MemSize := BlkCount; { Block count }
- MemSize := MemSize * BlkSize; { Current position }
- { * REMARK END * - Leon de Boer }
- End;
- ChangeListSize := True; { Successful }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TCollection OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TCollection--------------------------------------------------------------}
- { Init -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TCollection.Init (ALimit, ADelta: Sw_Integer);
- BEGIN
- Inherited Init; { Call ancestor }
- Delta := ADelta; { Set increment }
- SetLimit(ALimit); { Set limit }
- END;
- {--TCollection--------------------------------------------------------------}
- { Load -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TCollection.Load (Var S: TStream);
- VAR C, I: Sw_Integer;
- BEGIN
- If S.TPCompatible Then Begin
- { I ignore endianness issues here. If endianness is different,
- you can't expect binary compatible resources anyway. }
- Count := 0; S.Read(Count, Sizeof(Word));
- Limit := 0; S.Read(Limit, Sizeof(Word));
- Delta := 0; S.Read(Delta, Sizeof(Word))
- End
- Else Begin
- S.Read(Count, Sizeof(Count)); { Read count }
- S.Read(Limit, Sizeof(Limit)); { Read limit }
- S.Read(Delta, Sizeof(Delta)); { Read delta }
- End;
- Items := Nil; { Clear item pointer }
- C := Count; { Hold count }
- I := Limit; { Hold limit }
- Count := 0; { Clear count }
- Limit := 0; { Clear limit }
- SetLimit(I); { Set requested limit }
- Count := C; { Set count }
- For I := 0 To C-1 Do AtPut(I, GetItem(S)); { Get each item }
- END;
- {--TCollection--------------------------------------------------------------}
- { Done -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- DESTRUCTOR TCollection.Done;
- BEGIN
- FreeAll; { Free all items }
- SetLimit(0); { Release all memory }
- END;
- {--TCollection--------------------------------------------------------------}
- { At -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TCollection.At (Index: Sw_Integer): Pointer;
- BEGIN
- If (Index < 0) OR (Index >= Count) Then Begin { Invalid index }
- Error(coIndexError, Index); { Call error }
- At := Nil; { Return nil }
- End Else At := Items^[Index]; { Return item }
- END;
- {--TCollection--------------------------------------------------------------}
- { IndexOf -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TCollection.IndexOf (Item: Pointer): Sw_Integer;
- VAR I: Sw_Integer;
- BEGIN
- If (Count>0) Then Begin { Count is positive }
- For I := 0 To Count-1 Do { For each item }
- If (Items^[I]=Item) Then Begin { Look for match }
- IndexOf := I; { Return index }
- Exit; { Now exit }
- End;
- End;
- IndexOf := -1; { Return index }
- END;
- {--TCollection--------------------------------------------------------------}
- { GetItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TCollection.GetItem (Var S: TStream): Pointer;
- BEGIN
- GetItem := S.Get; { Item off stream }
- END;
- {--TCollection--------------------------------------------------------------}
- { LastThat -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TCollection.LastThat (Test: Pointer): Pointer;
- VAR I: LongInt;
- BEGIN
- For I := Count DownTo 1 Do
- Begin { Down from last item }
- IF Boolean(Byte(Longint(CallPointerLocal(Test,PreviousFramePointer,Items^[I-1])))) THEN
- Begin { Test each item }
- LastThat := Items^[I-1]; { Return item }
- Exit; { Now exit }
- End;
- End;
- LastThat := Nil; { None passed test }
- END;
- {--TCollection--------------------------------------------------------------}
- { FirstThat -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TCollection.FirstThat (Test: Pointer): Pointer;
- VAR I: LongInt;
- BEGIN
- For I := 1 To Count Do Begin { Up from first item }
- IF Boolean(Byte(Longint(CallPointerLocal(Test,PreviousFramePointer,Items^[I-1])))) THEN
- Begin { Test each item }
- FirstThat := Items^[I-1]; { Return item }
- Exit; { Now exit }
- End;
- End;
- FirstThat := Nil; { None passed test }
- END;
- {--TCollection--------------------------------------------------------------}
- { Pack -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.Pack;
- VAR I, J: Sw_Integer;
- BEGIN
- I := 0; { Initialize dest }
- J := 0; { Intialize test }
- While (I<Count) AND (J<Limit) Do Begin { Check fully packed }
- If (Items^[J]<>Nil) Then Begin { Found a valid item }
- If (I<>J) Then Begin
- Items^[I] := Items^[J]; { Transfer item }
- Items^[J] := Nil; { Now clear old item }
- End;
- Inc(I); { One item packed }
- End;
- Inc(J); { Next item to test }
- End;
- If (I<Count) Then Count := I; { New packed count }
- END;
- {--TCollection--------------------------------------------------------------}
- { FreeAll -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.FreeAll;
- VAR I: Sw_Integer;
- BEGIN
- for I := Count-1 downto 0 do
- FreeItem(At(I));
- Count := 0; { Clear item count }
- END;
- {--TCollection--------------------------------------------------------------}
- { DeleteAll -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.DeleteAll;
- BEGIN
- Count := 0; { Clear item count }
- END;
- {--TCollection--------------------------------------------------------------}
- { Free -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.Free (Item: Pointer);
- BEGIN
- Delete(Item); { Delete from list }
- FreeItem(Item); { Free the item }
- END;
- {--TCollection--------------------------------------------------------------}
- { Insert -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.Insert (Item: Pointer);
- BEGIN
- AtInsert(Count, Item); { Insert item }
- END;
- {--TCollection--------------------------------------------------------------}
- { Delete -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.Delete (Item: Pointer);
- BEGIN
- AtDelete(IndexOf(Item)); { Delete from list }
- END;
- {--TCollection--------------------------------------------------------------}
- { AtFree -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.AtFree (Index: Sw_Integer);
- VAR Item: Pointer;
- BEGIN
- Item := At(Index); { Retreive item ptr }
- AtDelete(Index); { Delete item }
- FreeItem(Item); { Free the item }
- END;
- {--TCollection--------------------------------------------------------------}
- { FreeItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.FreeItem (Item: Pointer);
- VAR P: PObject;
- BEGIN
- P := PObject(Item); { Convert pointer }
- If (P<>Nil) Then Dispose(P, Done); { Dispose of object }
- END;
- {--TCollection--------------------------------------------------------------}
- { AtDelete -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.AtDelete (Index: Sw_Integer);
- BEGIN
- If (Index >= 0) AND (Index < Count) Then Begin { Valid index }
- Dec(Count); { One less item }
- If (Count>Index) Then Move(Items^[Index+1],
- Items^[Index], (Count-Index)*Sizeof(Pointer)); { Shuffle items down }
- End Else Error(coIndexError, Index); { Index error }
- END;
- {--TCollection--------------------------------------------------------------}
- { ForEach -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.ForEach (Action: Pointer);
- VAR I: LongInt;
- BEGIN
- For I := 1 To Count Do { Up from first item }
- CallPointerLocal(Action,PreviousFramePointer,Items^[I-1]); { Call with each item }
- END;
- {--TCollection--------------------------------------------------------------}
- { SetLimit -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.SetLimit (ALimit: Sw_Integer);
- VAR
- AItems: PItemList;
- BEGIN
- If (ALimit < Count) Then
- ALimit := Count;
- If (ALimit > MaxCollectionSize) Then
- ALimit := MaxCollectionSize;
- If (ALimit <> Limit) Then
- Begin
- If (ALimit = 0) Then
- AItems := Nil
- Else
- Begin
- GetMem(AItems, ALimit * SizeOf(Pointer));
- If (AItems<>Nil) Then
- FillChar(AItems^,ALimit * SizeOf(Pointer), #0);
- End;
- If (AItems<>Nil) OR (ALimit=0) Then
- Begin
- If (AItems <>Nil) AND (Items <> Nil) Then
- Move(Items^, AItems^, Count*SizeOf(Pointer));
- If (Limit <> 0) AND (Items <> Nil) Then
- FreeMem(Items, Limit * SizeOf(Pointer));
- end;
- Items := AItems;
- Limit := ALimit;
- End;
- END;
- {--TCollection--------------------------------------------------------------}
- { Error -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.Error (Code, Info: Integer);
- BEGIN
- RunError(212 - Code); { Run error }
- END;
- {--TCollection--------------------------------------------------------------}
- { AtPut -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.AtPut (Index: Sw_Integer; Item: Pointer);
- BEGIN
- If (Index >= 0) AND (Index < Count) Then { Index valid }
- Items^[Index] := Item { Put item in index }
- Else Error(coIndexError, Index); { Index error }
- END;
- {--TCollection--------------------------------------------------------------}
- { AtInsert -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.AtInsert (Index: Sw_Integer; Item: Pointer);
- VAR I: Sw_Integer;
- BEGIN
- If (Index >= 0) AND (Index <= Count) Then Begin { Valid index }
- If (Count=Limit) Then SetLimit(Limit+Delta); { Expand size if able }
- If (Limit>Count) Then Begin
- If (Index < Count) Then Begin { Not last item }
- For I := Count-1 DownTo Index Do { Start from back }
- Items^[I+1] := Items^[I]; { Move each item }
- End;
- Items^[Index] := Item; { Put item in list }
- Inc(Count); { Inc count }
- End Else Error(coOverflow, Index); { Expand failed }
- End Else Error(coIndexError, Index); { Index error }
- END;
- {--TCollection--------------------------------------------------------------}
- { Store -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.Store (Var S: TStream);
- var
- LimitWord, DeltaWord: Word;
- PROCEDURE DoPutItem (P: Pointer);{$IFNDEF FPC}FAR;{$ENDIF}
- BEGIN
- PutItem(S, P); { Put item on stream }
- END;
- BEGIN
- If S.TPCompatible Then Begin
- { Check if it is safe to write in TP-compatible stream.
- If Count is too big, signal an error.
- If Limit or Delta are too big, write smaller values. }
- If (Count > MaxTPCompatibleCollectionSize)
- Then S.Error(stWriteError, 0)
- Else Begin
- S.Write(Count, Sizeof(Word));
- if Limit > MaxTPCompatibleCollectionSize
- then LimitWord := MaxTPCompatibleCollectionSize
- else LimitWord := Limit;
- S.Write(LimitWord, Sizeof(Word));
- if Delta > MaxTPCompatibleCollectionSize
- then DeltaWord := MaxTPCompatibleCollectionSize
- else DeltaWord := Delta;
- S.Write(DeltaWord, Sizeof(Word));
- End
- End
- Else Begin
- S.Write(Count, Sizeof(Count)); { Write count }
- S.Write(Limit, Sizeof(Limit)); { Write limit }
- S.Write(Delta, Sizeof(Delta)); { Write delta }
- End;
- ForEach(@DoPutItem); { Each item to stream }
- END;
- {--TCollection--------------------------------------------------------------}
- { PutItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TCollection.PutItem (Var S: TStream; Item: Pointer);
- BEGIN
- S.Put(Item); { Put item on stream }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TSortedCollection OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TSortedCollection--------------------------------------------------------}
- { Init -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TSortedCollection.Init (ALimit, ADelta: Sw_Integer);
- BEGIN
- Inherited Init(ALimit, ADelta); { Call ancestor }
- Duplicates := False; { Clear flag }
- END;
- {--TSortedCollection--------------------------------------------------------}
- { Load -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TSortedCollection.Load (Var S: TStream);
- BEGIN
- Inherited Load(S); { Call ancestor }
- S.Read(Duplicates, SizeOf(Duplicates)); { Read duplicate flag }
- END;
- {--TSortedCollection--------------------------------------------------------}
- { KeyOf -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TSortedCollection.KeyOf (Item: Pointer): Pointer;
- BEGIN
- KeyOf := Item; { Return item as key }
- END;
- {--TSortedCollection--------------------------------------------------------}
- { IndexOf -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TSortedCollection.IndexOf (Item: Pointer): Sw_Integer;
- VAR I, J: Sw_Integer;
- BEGIN
- J := -1; { Preset result }
- If Search(KeyOf(Item), I) Then Begin { Search for item }
- If Duplicates Then { Duplicates allowed }
- While (I < Count) AND (Item <> Items^[I]) Do
- Inc(I); { Count duplicates }
- If (I < Count) Then J := I; { Index result }
- End;
- IndexOf := J; { Return result }
- END;
- {--TSortedCollection--------------------------------------------------------}
- { Compare -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TSortedCollection.Compare (Key1, Key2: Pointer): Sw_Integer;
- BEGIN
- Abstract; { Abstract method }
- Compare:=0;
- END;
- {--TSortedCollection--------------------------------------------------------}
- { Search -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TSortedCollection.Search (Key: Pointer; Var Index: Sw_Integer): Boolean;
- VAR L, H, I, C: Sw_Integer;
- BEGIN
- Search := False; { Preset failure }
- L := 0; { Start count }
- H := Count - 1; { End count }
- While (L <= H) Do Begin
- I := (L + H) SHR 1; { Mid point }
- C := Compare(KeyOf(Items^[I]), Key); { Compare with key }
- If (C < 0) Then L := I + 1 Else Begin { Item to left }
- H := I - 1; { Item to right }
- If C = 0 Then Begin { Item match found }
- Search := True; { Result true }
- If NOT Duplicates Then L := I; { Force kick out }
- End;
- End;
- End;
- Index := L; { Return result }
- END;
- {--TSortedCollection--------------------------------------------------------}
- { Insert -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TSortedCollection.Insert (Item: Pointer);
- VAR I: Sw_Integer;
- BEGIN
- If NOT Search(KeyOf(Item), I) OR Duplicates Then { Item valid }
- AtInsert(I, Item); { Insert the item }
- END;
- {--TSortedCollection--------------------------------------------------------}
- { Store -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TSortedCollection.Store (Var S: TStream);
- BEGIN
- TCollection.Store(S); { Call ancestor }
- S.Write(Duplicates, SizeOf(Duplicates)); { Write duplicate flag }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TStringCollection OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TStringCollection--------------------------------------------------------}
- { GetItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TStringCollection.GetItem (Var S: TStream): Pointer;
- BEGIN
- GetItem := S.ReadStr; { Get new item }
- END;
- {--TStringCollection--------------------------------------------------------}
- { Compare -> Platforms DOS/DPMI/WIN/OS2 - Checked 21Aug97 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TStringCollection.Compare (Key1, Key2: Pointer): Sw_Integer;
- VAR I, J: Sw_Integer; P1, P2: PString;
- BEGIN
- P1 := PString(Key1); { String 1 pointer }
- P2 := PString(Key2); { String 2 pointer }
- If (Length(P1^)<Length(P2^)) Then J := Length(P1^)
- Else J := Length(P2^); { Shortest length }
- I := 1; { First character }
- While (I<J) AND (P1^[I]=P2^[I]) Do Inc(I); { Scan till fail }
- If (I=J) Then Begin { Possible match }
- { * REMARK * - Bug fix 21 August 1997 }
- If (P1^[I]<P2^[I]) Then Compare := -1 Else { String1 < String2 }
- If (P1^[I]>P2^[I]) Then Compare := 1 Else { String1 > String2 }
- If (Length(P1^)>Length(P2^)) Then Compare := 1 { String1 > String2 }
- Else If (Length(P1^)<Length(P2^)) Then { String1 < String2 }
- Compare := -1 Else Compare := 0; { String1 = String2 }
- { * REMARK END * - Leon de Boer }
- End Else If (P1^[I]<P2^[I]) Then Compare := -1 { String1 < String2 }
- Else Compare := 1; { String1 > String2 }
- END;
- {--TStringCollection--------------------------------------------------------}
- { FreeItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStringCollection.FreeItem (Item: Pointer);
- BEGIN
- DisposeStr(Item); { Dispose item }
- END;
- {--TStringCollection--------------------------------------------------------}
- { PutItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 22May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStringCollection.PutItem (Var S: TStream; Item: Pointer);
- BEGIN
- S.WriteStr(Item); { Write string }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TStrCollection OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TStrCollection-----------------------------------------------------------}
- { Compare -> Platforms DOS/DPMI/WIN/OS2 - Checked 23May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TStrCollection.Compare (Key1, Key2: Pointer): Sw_Integer;
- VAR I, J: Sw_Integer; P1, P2: PByteArray;
- BEGIN
- P1 := PByteArray(Key1); { PChar 1 pointer }
- P2 := PByteArray(Key2); { PChar 2 pointer }
- I := 0; { Preset no size }
- If (P1<>Nil) Then While (P1^[I]<>0) Do Inc(I); { PChar 1 length }
- J := 0; { Preset no size }
- If (P2<>Nil) Then While (P2^[J]<>0) Do Inc(J); { PChar 2 length }
- If (I < J) Then J := I; { Shortest length }
- I := 0; { First character }
- While (I<J) AND (P1^[I]=P2^[I]) Do Inc(I); { Scan till fail }
- If (P1^[I]=P2^[I]) Then Compare := 0 Else { Strings matched }
- If (P1^[I]<P2^[I]) Then Compare := -1 Else { String1 < String2 }
- Compare := 1; { String1 > String2 }
- END;
- {--TStrCollection-----------------------------------------------------------}
- { GetItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 23May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TStrCollection.GetItem (Var S: TStream): Pointer;
- BEGIN
- GetItem := S.StrRead; { Get string item }
- END;
- {--TStrCollection-----------------------------------------------------------}
- { FreeItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 23May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStrCollection.FreeItem (Item: Pointer);
- VAR I: Sw_Integer; P: PByteArray;
- BEGIN
- If (Item<>Nil) Then Begin { Item is valid }
- P := PByteArray(Item); { Create byte pointer }
- I := 0; { Preset no size }
- While (P^[I]<>0) Do Inc(I); { Find PChar end }
- FreeMem(Item, I+1); { Release memory }
- End;
- END;
- {--TStrCollection-----------------------------------------------------------}
- { PutItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 23May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStrCollection.PutItem (Var S: TStream; Item: Pointer);
- BEGIN
- S.StrWrite(Item); { Write the string }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TUnSortedStrCollection OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TUnSortedCollection------------------------------------------------------}
- { Insert -> Platforms DOS/DPMI/WIN/OS2 - Checked 23May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TUnSortedStrCollection.Insert (Item: Pointer);
- BEGIN
- AtInsert(Count, Item); { Insert - NO sorting }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TResourceItem RECORD }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- TYPE
- TResourceItem = packed RECORD
- Posn: LongInt; { Resource position }
- Size: LongInt; { Resource size }
- Key : String; { Resource key }
- End;
- PResourceItem = ^TResourceItem;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TResourceCollection OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TResourceCollection------------------------------------------------------}
- { KeyOf -> Platforms DOS/DPMI/WIN/OS2 - Checked 24May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TResourceCollection.KeyOf (Item: Pointer): Pointer;
- BEGIN
- KeyOf := @PResourceItem(Item)^.Key; { Pointer to key }
- END;
- {--TResourceCollection------------------------------------------------------}
- { GetItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 24May96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TResourceCollection.GetItem (Var S: TStream): Pointer;
- VAR B: Byte; Pos: Longint; Size: Longint; Ts: String; P: PResourceItem;
- BEGIN
- S.Read(Pos, SizeOf(Pos)); { Read position }
- S.Read(Size, SizeOf(Size)); { Read size }
- S.Read(B, 1); { Read key length }
- GetMem(P, B + (SizeOf(TResourceItem) -
- SizeOf(Ts) + 1)); { Allocate min memory }
- If (P<>Nil) Then Begin { If allocate works }
- P^.Posn := Pos; { Xfer position }
- P^.Size := Size; { Xfer size }
- P^.Key[0] := Char(B); { Xfer string length }
- S.Read(P^.Key[1], B); { Xfer string data }
- End;
- GetItem := P; { Return pointer }
- END;
- {--TResourceCollection------------------------------------------------------}
- { FreeItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 24May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TResourceCollection.FreeItem (Item: Pointer);
- VAR Ts: String;
- BEGIN
- If (Item<>Nil) Then FreeMem(Item,
- SizeOf(TResourceItem) - SizeOf(Ts) +
- Length(PResourceItem(Item)^.Key) + 1); { Release memory }
- END;
- {--TResourceCollection------------------------------------------------------}
- { PutItem -> Platforms DOS/DPMI/WIN/OS2 - Checked 24May96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TResourceCollection.PutItem (Var S: TStream; Item: Pointer);
- VAR Ts: String;
- BEGIN
- If (Item<>Nil) Then S.Write(PResourceItem(Item)^,
- SizeOf(TResourceItem) - SizeOf(Ts) +
- Length(PResourceItem(Item)^.Key) + 1); { Write to stream }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { PRIVATE RESOURCE MANAGER CONSTANTS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- CONST
- RStreamMagic: LongInt = $52504246; { 'FBPR' }
- RStreamBackLink: LongInt = $4C424246; { 'FBBL' }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { PRIVATE RESOURCE MANAGER TYPES }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- TYPE
- {$IFDEF NewExeFormat} { New EXE format }
- TExeHeader = packed RECORD
- eHdrSize: Word;
- eMinAbove: Word;
- eMaxAbove: Word;
- eInitSS: Word;
- eInitSP: Word;
- eCheckSum: Word;
- eInitPC: Word;
- eInitCS: Word;
- eRelocOfs: Word;
- eOvlyNum: Word;
- eRelocTab: Word;
- eSpace: Array[1..30] of Byte;
- eNewHeader: Word;
- END;
- {$ENDIF}
- THeader = packed RECORD
- Signature: Word;
- Case Integer Of
- 0: (
- LastCount: Word;
- PageCount: Word;
- ReloCount: Word);
- 1: (
- InfoType: Word;
- InfoSize: Longint);
- End;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TResourceFile OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TResourceFile------------------------------------------------------------}
- { Init -> Platforms DOS/DPMI/WIN/OS2 - Checked 18Jun96 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TResourceFile.Init(AStream: PStream);
- VAR Found, Stop: Boolean; Header: THeader;
- {$IFDEF NewExeFormat} ExeHeader: TExeHeader; {$ENDIF}
- BEGIN
- TObject.Init; { Initialize object }
- Found := False; { Preset false }
- If (AStream<>Nil) Then Begin
- Stream := AStream; { Hold stream }
- BasePos := Stream^.GetPos; { Get position }
- Repeat
- Stop := True; { Preset stop }
- If (BasePos <= Stream^.GetSize-SizeOf(THeader))
- Then Begin { Valid file header }
- Stream^.Seek(BasePos); { Seek to position }
- Stream^.Read(Header, SizeOf(THeader)); { Read header }
- Case Header.Signature Of
- {$IFDEF NewExeFormat} { New format file }
- $5A4D: Begin
- Stream^.Read(ExeHeader, SizeOf(TExeHeader));
- BasePos := ExeHeader.eNewHeader; { Hold position }
- Stop := False; { Clear stop flag }
- End;
- $454E: Begin
- BasePos := Stream^.GetSize - 8; { Hold position }
- Stop := False; { Clear stop flag }
- End;
- $4246: Begin
- Stop := False; { Clear stop flag }
- Case Header.Infotype Of
- $5250: Begin { Found Resource }
- Found := True; { Found flag is true }
- Stop := True; { Set stop flag }
- End;
- $4C42: Dec(BasePos, Header.InfoSize-8);{ Found BackLink }
- $4648: Dec(BasePos, SizeOf(THeader)*2);{ Found HelpFile }
- Else Stop := True; { Set stop flag }
- End;
- End;
- $424E: If Header.InfoType = $3230 { Found Debug Info }
- Then Begin
- Dec(BasePos, Header.InfoSize); { Adjust position }
- Stop := False; { Clear stop flag }
- End;
- {$ELSE}
- $5A4D: Begin
- Inc(BasePos, LongInt(Header.PageCount)*512
- - (-Header.LastCount AND 511)); { Calc position }
- Stop := False; { Clear stop flag }
- End;
- $4246: If Header.InfoType = $5250 Then { Header was found }
- Found := True Else Begin
- Inc(BasePos, Header.InfoSize + 8); { Adjust position }
- Stop := False; { Clear stop flag }
- End;
- {$ENDIF}
- End;
- End;
- Until Stop; { Until flag is set }
- End;
- If Found Then Begin { Resource was found }
- Stream^.Seek(BasePos + SizeOf(LongInt) * 2); { Seek to position }
- Stream^.Read(IndexPos, SizeOf(LongInt)); { Read index position }
- Stream^.Seek(BasePos + IndexPos); { Seek to resource }
- Index.Load(Stream^); { Load resource }
- End Else Begin
- IndexPos := SizeOf(LongInt) * 3; { Set index position }
- Index.Init(0, 8); { Set index }
- End;
- END;
- {--TResourceFile------------------------------------------------------------}
- { Done -> Platforms DOS/DPMI/WIN/OS2 - Checked 18Jun96 LdB }
- {---------------------------------------------------------------------------}
- DESTRUCTOR TResourceFile.Done;
- BEGIN
- Flush; { Flush the file }
- Index.Done; { Dispose of index }
- If (Stream<>Nil) Then Dispose(Stream, Done); { Dispose of stream }
- END;
- {--TResourceFile------------------------------------------------------------}
- { Count -> Platforms DOS/DPMI/WIN/OS2 - Checked 18Jun96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TResourceFile.Count: Sw_Integer;
- BEGIN
- Count := Index.Count; { Return index count }
- END;
- {--TResourceFile------------------------------------------------------------}
- { KeyAt -> Platforms DOS/DPMI/WIN/OS2 - Checked 18Jun96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TResourceFile.KeyAt (I: Sw_Integer): String;
- BEGIN
- KeyAt := PResourceItem(Index.At(I))^.Key; { Return key }
- END;
- {--TResourceFile------------------------------------------------------------}
- { Get -> Platforms DOS/DPMI/WIN/OS2 - Checked 18Jun96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TResourceFile.Get (Key: String): PObject;
- VAR I: Sw_Integer;
- BEGIN
- If (Stream = Nil) OR (NOT Index.Search(@Key, I)) { No match on key }
- Then Get := Nil Else Begin
- Stream^.Seek(BasePos +
- PResourceItem(Index.At(I))^.Posn); { Seek to position }
- Get := Stream^.Get; { Get item }
- End;
- END;
- {--TResourceFile------------------------------------------------------------}
- { SwitchTo -> Platforms DOS/DPMI/WIN/OS2 - Checked 18Jun96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TResourceFile.SwitchTo (AStream: PStream; Pack: Boolean): PStream;
- VAR NewBasePos: LongInt;
- PROCEDURE DoCopyResource (Item: PResourceItem);{$IFNDEF FPC}FAR;{$ENDIF}
- BEGIN
- Stream^.Seek(BasePos + Item^.Posn); { Move stream position }
- Item^.Posn := AStream^.GetPos - NewBasePos; { Hold new position }
- AStream^.CopyFrom(Stream^, Item^.Size); { Copy the item }
- END;
- BEGIN
- SwitchTo := Stream; { Preset return }
- If (AStream<>Nil) AND (Stream<>Nil) Then Begin { Both streams valid }
- NewBasePos := AStream^.GetPos; { Get position }
- If Pack Then Begin
- AStream^.Seek(NewBasePos + SizeOf(LongInt)*3); { Seek to position }
- Index.ForEach(@DoCopyResource); { Copy each resource }
- IndexPos := AStream^.GetPos - NewBasePos; { Hold index position }
- End Else Begin
- Stream^.Seek(BasePos); { Seek to position }
- AStream^.CopyFrom(Stream^, IndexPos); { Copy the resource }
- End;
- Stream := AStream; { Hold new stream }
- BasePos := NewBasePos; { New base position }
- Modified := True; { Set modified flag }
- End;
- END;
- {--TResourceFile------------------------------------------------------------}
- { Flush -> Platforms DOS/DPMI/WIN/OS2 - Checked 18Jun96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TResourceFile.Flush;
- VAR ResSize: LongInt; LinkSize: LongInt;
- BEGIN
- If (Modified) AND (Stream<>Nil) Then Begin { We have modification }
- Stream^.Seek(BasePos + IndexPos); { Seek to position }
- Index.Store(Stream^); { Store the item }
- ResSize := Stream^.GetPos - BasePos; { Hold position }
- LinkSize := ResSize + SizeOf(LongInt) * 2; { Hold link size }
- Stream^.Write(RStreamBackLink, SizeOf(LongInt)); { Write link back }
- Stream^.Write(LinkSize, SizeOf(LongInt)); { Write link size }
- Stream^.Seek(BasePos); { Move stream position }
- Stream^.Write(RStreamMagic, SizeOf(LongInt)); { Write number }
- Stream^.Write(ResSize, SizeOf(LongInt)); { Write record size }
- Stream^.Write(IndexPos, SizeOf(LongInt)); { Write index position }
- Stream^.Flush; { Flush the stream }
- End;
- Modified := False; { Clear modified flag }
- END;
- {--TResourceFile------------------------------------------------------------}
- { Delete -> Platforms DOS/DPMI/WIN/OS2 - Checked 18Jun96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TResourceFile.Delete (Key: String);
- VAR I: Sw_Integer;
- BEGIN
- If Index.Search(@Key, I) Then Begin { Search for key }
- Index.Free(Index.At(I)); { Delete from index }
- Modified := True; { Set modified flag }
- End;
- END;
- {--TResourceFile------------------------------------------------------------}
- { Put -> Platforms DOS/DPMI/WIN/OS2 - Checked 18Jun96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TResourceFile.Put (Item: PObject; Key: String);
- VAR I: Sw_Integer; Ts: String; P: PResourceItem;
- BEGIN
- If (Stream=Nil) Then Exit; { Stream not valid }
- If Index.Search(@Key, I) Then P := Index.At(I) { Search for item }
- Else Begin
- GetMem(P, Length(Key) + (SizeOf(TResourceItem) -
- SizeOf(Ts) + 1)); { Allocate memory }
- If (P<>Nil) Then Begin
- P^.Key := Key; { Store key }
- Index.AtInsert(I, P); { Insert item }
- End;
- End;
- If (P<>Nil) Then Begin
- P^.Posn := IndexPos; { Set index position }
- Stream^.Seek(BasePos + IndexPos); { Seek file position }
- Stream^.Put(Item); { Put item on stream }
- IndexPos := Stream^.GetPos - BasePos; { Hold index position }
- P^.Size := IndexPos - P^.Posn; { Calc size }
- Modified := True; { Set modified flag }
- End;
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TStringList OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TStringList--------------------------------------------------------------}
- { Load -> Platforms DOS/DPMI/WIN/OS2 - Checked 30Jun97 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TStringList.Load (Var S: TStream);
- VAR Size: Word;
- BEGIN
- Stream := @S; { Hold stream pointer }
- S.Read(Size, SizeOf(Word)); { Read size }
- BasePos := S.GetPos; { Hold position }
- S.Seek(BasePos + Size); { Seek to position }
- S.Read(IndexSize, SizeOf(Integer)); { Read index size }
- GetMem(Index, IndexSize * SizeOf(TStrIndexRec)); { Allocate memory }
- S.Read(Index^, IndexSize * SizeOf(TStrIndexRec)); { Read indexes }
- END;
- {--TStringList--------------------------------------------------------------}
- { Done -> Platforms DOS/DPMI/WIN/OS2 - Checked 30Jun97 LdB }
- {---------------------------------------------------------------------------}
- DESTRUCTOR TStringList.Done;
- BEGIN
- FreeMem(Index, IndexSize * SizeOf(TStrIndexRec)); { Release memory }
- END;
- {--TStringList--------------------------------------------------------------}
- { Get -> Platforms DOS/DPMI/WIN/OS2 - Checked 30Jun97 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION TStringList.Get (Key: Sw_Word): String;
- VAR I: Word; S: String;
- BEGIN
- S := ''; { Preset empty string }
- If (IndexSize>0) Then Begin { We must have strings }
- I := 0; { First entry }
- While (I<IndexSize) AND (S='') Do Begin
- If ((Key - Index^[I].Key)<Index^[I].Count) { Diff less than count }
- Then ReadStr(S, Index^[I].Offset,
- Key-Index^[I].Key); { Read the string }
- Inc(I); { Next entry }
- End;
- End;
- Get := S; { Return empty string }
- END;
- {***************************************************************************}
- { TStringList PRIVATE METHODS }
- {***************************************************************************}
- {--TStringLis---------------------------------------------------------------}
- { ReadStr -> Platforms DOS/DPMI/WIN/OS2 - Checked 30Jun97 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStringList.ReadStr (Var S: String; Offset, Skip: Sw_Word);
- BEGIN
- Stream^.Seek(BasePos + Offset); { Seek to position }
- Inc(Skip); { Adjust skip }
- Repeat
- Stream^.Read(S[0], 1); { Read string size }
- Stream^.Read(S[1], Ord(S[0])); { Read string data }
- Dec(Skip); { One string read }
- Until (Skip = 0);
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { TStrListMaker OBJECT METHODS }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {--TStrListMaker------------------------------------------------------------}
- { Init -> Platforms DOS/DPMI/WIN/OS2 - Checked 30Jun97 LdB }
- {---------------------------------------------------------------------------}
- CONSTRUCTOR TStrListMaker.Init (AStrSize, AIndexSize: Sw_Word);
- BEGIN
- Inherited Init; { Call ancestor }
- StrSize := AStrSize; { Hold size }
- IndexSize := AIndexSize; { Hold index size }
- GetMem(Strings, AStrSize); { Allocate memory }
- GetMem(Index, AIndexSize * SizeOf(TStrIndexRec)); { Allocate memory }
- END;
- {--TStrListMaker------------------------------------------------------------}
- { Done -> Platforms DOS/DPMI/WIN/OS2 - Checked 30Jun97 LdB }
- {---------------------------------------------------------------------------}
- DESTRUCTOR TStrListMaker.Done;
- BEGIN
- FreeMem(Index, IndexSize * SizeOf(TStrIndexRec)); { Free index memory }
- FreeMem(Strings, StrSize); { Free data memory }
- END;
- {--TStrListMaker------------------------------------------------------------}
- { Put -> Platforms DOS/DPMI/WIN/OS2 - Checked 30Jun97 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStrListMaker.Put (Key: Sw_Word; S: String);
- BEGIN
- If (Cur.Count = 16) OR (Key <> Cur.Key + Cur.Count)
- Then CloseCurrent; { Close current }
- If (Cur.Count = 0) Then Begin
- Cur.Key := Key; { Set key }
- Cur.Offset := StrPos; { Set offset }
- End;
- Inc(Cur.Count); { Inc count }
- Move(S, Strings^[StrPos], Length(S) + 1); { Move string data }
- Inc(StrPos, Length(S) + 1); { Adjust position }
- END;
- {--TStrListMaker------------------------------------------------------------}
- { Store -> Platforms DOS/DPMI/WIN/OS2 - Checked 30Jun97 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStrListMaker.Store (Var S: TStream);
- BEGIN
- CloseCurrent; { Close all current }
- S.Write(StrPos, SizeOf(Word)); { Write position }
- S.Write(Strings^, StrPos); { Write string data }
- S.Write(IndexPos, SizeOf(Word)); { Write index position }
- S.Write(Index^, IndexPos * SizeOf(TStrIndexRec)); { Write indexes }
- END;
- {***************************************************************************}
- { TStrListMaker PRIVATE METHODS }
- {***************************************************************************}
- {--TStrListMaker------------------------------------------------------------}
- { CloseCurrent -> Platforms DOS/DPMI/WIN/OS2 - Checked 30Jun97 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE TStrListMaker.CloseCurrent;
- BEGIN
- If (Cur.Count <> 0) Then Begin
- Index^[IndexPos] := Cur; { Hold index position }
- Inc(IndexPos); { Next index }
- Cur.Count := 0; { Adjust count }
- End;
- END;
- {***************************************************************************}
- { INTERFACE ROUTINES }
- {***************************************************************************}
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { DYNAMIC STRING INTERFACE ROUTINES }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {---------------------------------------------------------------------------}
- { NewStr -> Platforms DOS/DPMI/WINDOWS/OS2 - Checked 12Jun96 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION NewStr (Const S: String): PString;
- VAR P: PString;
- BEGIN
- If (S = '') Then P := Nil Else Begin { Return nil }
- GetMem(P, Length(S) + 1); { Allocate memory }
- If (P<>Nil) Then P^ := S; { Hold string }
- End;
- NewStr := P; { Return result }
- END;
- {---------------------------------------------------------------------------}
- { DisposeStr -> Platforms DOS/DPMI/WINDOWS/OS2 - Checked 12Jun96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE DisposeStr (P: PString);
- BEGIN
- If (P <> Nil) Then FreeMem(P, Length(P^) + 1); { Release memory }
- END;
- PROCEDURE SetStr(VAR p:pString; CONST s:STRING);
- BEGIN
- IF p<>NIL THEN
- FreeMem(P, Length(P^) + 1);
- GetMem(p,LENGTH(s)+1);
- pSTRING(p)^ := s
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { STREAM INTERFACE ROUTINES }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {---------------------------------------------------------------------------}
- { Abstract -> Platforms DOS/DPMI/WINDOWS/OS2 - Checked 12Jun96 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE Abstract;
- BEGIN
- RunError(211); { Abstract error }
- END;
- {---------------------------------------------------------------------------}
- { RegisterObjects -> Platforms DOS/DPMI/WINDOWS/OS2 - Checked 02Sep97 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE RegisterObjects;
- BEGIN
- RegisterType(RCollection); { Register object }
- RegisterType(RStringCollection); { Register object }
- RegisterType(RStrCollection); { Register object }
- END;
- {---------------------------------------------------------------------------}
- { RegisterType -> Platforms DOS/DPMI/WINDOWS/OS2 - Checked 02Sep97 LdB }
- {---------------------------------------------------------------------------}
- PROCEDURE RegisterType (Var S: TStreamRec);
- VAR P: PStreamRec;
- BEGIN
- P := StreamTypes; { Current reg list }
- While (P <> Nil) AND (P^.ObjType <> S.ObjType)
- Do P := P^.Next; { Find end of chain }
- If (P = Nil) AND (S.ObjType <> 0) Then Begin { Valid end found }
- S.Next := StreamTypes; { Chain the list }
- StreamTypes := @S; { We are now first }
- End Else RegisterError; { Register the error }
- END;
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- { GENERAL FUNCTION INTERFACE ROUTINES }
- {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
- {---------------------------------------------------------------------------}
- { LongMul -> Platforms DOS/DPMI/WINDOWS/OS2 - Checked 04Sep97 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION LongMul (X, Y: Integer): LongInt;
- BEGIN
- LongMul:=Longint(X*Y);
- END;
- {---------------------------------------------------------------------------}
- { LongDiv -> Platforms DOS/DPMI/WINDOWS/OS2 - Checked 04Sep97 LdB }
- {---------------------------------------------------------------------------}
- FUNCTION LongDiv (X: LongInt; Y: Integer): Integer;
- BEGIN
- LongDiv := Integer(X DIV Y);
- END;
- END.
- {
- $Log$
- Revision 1.18 2003-01-05 16:27:05 hajny
- * PString inherited from System
- Revision 1.17 2002/12/07 14:37:15 carl
- - avoid warnings (add typecast)
- Revision 1.16 2002/10/31 13:08:04 carl
- * forgot to init P buffer variable = crash
- Revision 1.15 2002/10/31 12:47:30 carl
- * more compatibility fixes for objects unit
- Revision 1.14 2002/10/30 22:44:44 carl
- * Bugfix for error checking
- - DosStreamError is no longer global (bugfix 2043)
- Revision 1.13 2002/10/09 16:10:14 carl
- - removed OS depedencies
- Revision 1.12 2002/09/07 21:15:25 carl
- - some cleanup (less ifdef)
- Revision 1.11 2002/09/07 15:07:45 peter
- * old logs removed and tabs fixed
- Revision 1.10 2002/09/01 19:03:19 florian
- + powerpc support added
- Revision 1.9 2002/03/28 20:56:00 carl
- - remove go32v1 support
- }
|