| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788 | {    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by Florian Klaempfl    member of the Free Pascal development team    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. **********************************************************************}{ This unit provides the same Functionality as the TypInfo Unit }{ of Delphi                                                     }unit TypInfo;  interface{$MODE objfpc}{$MODESWITCH AdvancedRecords}{$inline on}{$macro on}{$h+}  uses SysUtils;// temporary types:    type{$MINENUMSIZE 1   this saves a lot of memory }{$ifdef FPC_RTTI_PACKSET1}{ for Delphi compatibility }{$packset 1}{$endif}       { this alias and the following constant aliases are for backwards         compatibility before TTypeKind was moved to System unit }       TTypeKind = System.TTypeKind;    const       tkUnknown = System.tkUnknown;       tkInteger = System.tkInteger;       tkChar = System.tkChar;       tkEnumeration = System.tkEnumeration;       tkFloat = System.tkFloat;       tkSet = System.tkSet;       tkMethod = System.tkMethod;       tkSString = System.tkSString;       tkLString = System.tkLString;       tkAString = System.tkAString;       tkWString = System.tkWString;       tkVariant = System.tkVariant;       tkArray = System.tkArray;       tkRecord = System.tkRecord;       tkInterface = System.tkInterface;       tkClass = System.tkClass;       tkObject = System.tkObject;       tkWChar = System.tkWChar;       tkBool = System.tkBool;       tkInt64 = System.tkInt64;       tkQWord = System.tkQWord;       tkDynArray = System.tkDynArray;       tkInterfaceRaw = System.tkInterfaceRaw;       tkProcVar = System.tkProcVar;       tkUString = System.tkUString;       tkUChar = System.tkUChar;       tkHelper = System.tkHelper;       tkFile = System.tkFile;       tkClassRef = System.tkClassRef;       tkPointer = System.tkPointer;    type       TOrdType  = (otSByte,otUByte,otSWord,otUWord,otSLong,otULong,otSQWord,otUQWord);{$ifndef FPUNONE}       TFloatType = (ftSingle,ftDouble,ftExtended,ftComp,ftCurr);{$endif}       TMethodKind = (mkProcedure,mkFunction,mkConstructor,mkDestructor,                      mkClassProcedure,mkClassFunction,mkClassConstructor,                      mkClassDestructor,mkOperatorOverload);       TParamFlag     = (pfVar,pfConst,pfArray,pfAddress,pfReference,pfOut,pfConstRef                         {$ifndef VER3_0},pfHidden,pfHigh,pfSelf,pfVmt,pfResult{$endif VER3_0}                         );       TParamFlags    = set of TParamFlag;       TIntfFlag      = (ifHasGuid,ifDispInterface,ifDispatch,ifHasStrGUID);       TIntfFlags     = set of TIntfFlag;       TIntfFlagsBase = set of TIntfFlag;       // don't rely on integer values of TCallConv since it includes all conventions       // which both Delphi and FPC support. In the future Delphi can support more and       // FPC's own conventions will be shifted/reordered accordingly       TCallConv = (ccReg, ccCdecl, ccPascal, ccStdCall, ccSafeCall,                    ccCppdecl, ccFar16, ccOldFPCCall, ccInternProc,                    ccSysCall, ccSoftFloat, ccMWPascal);{$push}{$scopedenums on}       TSubRegister = (         None,         Lo,         Hi,         Word,         DWord,         QWord,         FloatSingle,         FloatDouble,         FloatQuad,         MultiMediaSingle,         MultiMediaDouble,         MultiMediaWhole,         MultiMediaX,         MultiMediaY       );       TRegisterType = (         Invalid,         Int,         FP,         MMX,         MultiMedia,         Special,         Address       );{$pop}{$MINENUMSIZE DEFAULT}   const      ptField = 0;      ptStatic = 1;      ptVirtual = 2;      ptConst = 3;   type      TTypeKinds = set of TTypeKind;      ShortStringBase = string[255];      PParameterLocation = ^TParameterLocation;      TParameterLocation ={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed{$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        LocType: Byte;        function GetRegType: TRegisterType; inline;        function GetReference: Boolean; inline;        function GetShiftVal: Int8; inline;      public        RegSub: TSubRegister;        RegNumber: Word;        { Stack offset if Reference, ShiftVal if not }        Offset: SizeInt;        { if Reference then the register is the index register otherwise the          register in wihch (part of) the parameter resides }        property Reference: Boolean read GetReference;        property RegType: TRegisterType read GetRegType;        { if Reference, otherwise 0 }        property ShiftVal: Int8 read GetShiftVal;      end;      PParameterLocations = ^TParameterLocations;      TParameterLocations ={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed{$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetLocation(aIndex: Byte): PParameterLocation; inline;        function GetTail: Pointer; inline;      public        Count: Byte;        property Location[Index: Byte]: PParameterLocation read GetLocation;        property Tail: Pointer read GetTail;      end;      PVmtFieldClassTab = ^TVmtFieldClassTab;      TVmtFieldClassTab ={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed{$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record        Count: Word;        ClassRef: array[0..0] of PClass;      end;      PVmtFieldEntry = ^TVmtFieldEntry;      TVmtFieldEntry ={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed{$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetNext: PVmtFieldEntry; inline;        function GetTail: Pointer; inline;      public        FieldOffset: PtrUInt;        TypeIndex: Word;        Name: ShortString;        property Tail: Pointer read GetTail;        property Next: PVmtFieldEntry read GetNext;      end;      PVmtFieldTable = ^TVmtFieldTable;      TVmtFieldTable ={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed{$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetField(aIndex: Word): PVmtFieldEntry;      public        Count: Word;        ClassTab: PVmtFieldClassTab;        { should be array[Word] of TFieldInfo;  but          Elements have variant size! force at least proper alignment }        Fields: array[0..0] of TVmtFieldEntry;        property Field[aIndex: Word]: PVmtFieldEntry read GetField;      end;{$PACKRECORDS 1}      TTypeInfo = record         Kind : TTypeKind;         Name : ShortString;         // here the type data follows as TTypeData record      end;      PTypeInfo = ^TTypeInfo;      PPTypeInfo = ^PTypeInfo;      PPropData = ^TPropData;{ Note: these are only for backwards compatibility. New type references should        only use PPTypeInfo directly! }{$ifdef ver3_0}{$define TypeInfoPtr := PTypeInfo}{$else}{$define TypeInfoPtr := PPTypeInfo}{$endif}{$PACKRECORDS C}{$if not defined(VER3_0) and not defined(VER3_2)}{$define PROVIDE_ATTR_TABLE}{$endif}      TAttributeProc = function : TCustomAttribute;      TAttributeEntry =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif}      record        AttrType: PPTypeInfo;        AttrCtor: CodePointer;        AttrProc: TAttributeProc;        ArgLen: Word;        ArgData: Pointer;      end;{$ifdef CPU16}      TAttributeEntryList = array[0..(High(SizeUInt) div SizeOf(TAttributeEntry))-1] of TAttributeEntry;{$else CPU16}      TAttributeEntryList = array[0..$ffff] of TAttributeEntry;{$endif CPU16}      TAttributeTable =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif}      record        AttributeCount: word;        AttributesList: TAttributeEntryList;      end;      PAttributeTable = ^TAttributeTable;      // members of TTypeData      TArrayTypeData ={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed{$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetElType: PTypeInfo; inline;        function GetDims(aIndex: Byte): PTypeInfo; inline;      public        property ElType: PTypeInfo read GetElType;        property Dims[Index: Byte]: PTypeInfo read GetDims;      public        Size: SizeInt;        ElCount: SizeInt;        ElTypeRef: TypeInfoPtr;        DimCount: Byte;        DimsRef: array[0..255] of TypeInfoPtr;      end;      PManagedField = ^TManagedField;      TManagedField =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetTypeRef: PTypeInfo; inline;      public        property TypeRef: PTypeInfo read GetTypeRef;      public        TypeRefRef: TypeInfoPtr;        FldOffset: SizeInt;      end;      PInitManagedField = ^TInitManagedField;      TInitManagedField = TManagedField;      PProcedureParam = ^TProcedureParam;      TProcedureParam =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetParamType: PTypeInfo; inline;        function GetFlags: Byte; inline;      public        property ParamType: PTypeInfo read GetParamType;        property Flags: Byte read GetFlags;      public        ParamFlags: TParamFlags;        ParamTypeRef: TypeInfoPtr;        Name: ShortString;      end;      TProcedureSignature =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetResultType: PTypeInfo; inline;      public        property ResultType: PTypeInfo read GetResultType;      public        Flags: Byte;        CC: TCallConv;        ResultTypeRef: TypeInfoPtr;        ParamCount: Byte;        {Params: array[0..ParamCount - 1] of TProcedureParam;}        function GetParam(ParamIndex: Integer): PProcedureParam;      end;      PVmtMethodParam = ^TVmtMethodParam;      TVmtMethodParam =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetTail: Pointer; inline;        function GetNext: PVmtMethodParam; inline;        function GetName: ShortString; inline;      public        ParamType: PPTypeInfo;        Flags: TParamFlags;        NamePtr: PShortString;        ParaLocs: PParameterLocations;        property Name: ShortString read GetName;        property Tail: Pointer read GetTail;        property Next: PVmtMethodParam read GetNext;      end;      PIntfMethodEntry = ^TIntfMethodEntry;      TIntfMethodEntry =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetParam(Index: Word): PVmtMethodParam;        function GetResultLocs: PParameterLocations; inline;        function GetTail: Pointer; inline;        function GetNext: PIntfMethodEntry; inline;        function GetName: ShortString; inline;      public        ResultType: PPTypeInfo;        CC: TCallConv;        Kind: TMethodKind;        ParamCount: Word;        StackSize: SizeInt;        NamePtr: PShortString;        { Params: array[0..ParamCount - 1] of TVmtMethodParam }        { ResultLocs: PParameterLocations (if ResultType != Nil) }        property Name: ShortString read GetName;        property Param[Index: Word]: PVmtMethodParam read GetParam;        property ResultLocs: PParameterLocations read GetResultLocs;        property Tail: Pointer read GetTail;        property Next: PIntfMethodEntry read GetNext;      end;      PIntfMethodTable = ^TIntfMethodTable;      TIntfMethodTable =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetMethod(Index: Word): PIntfMethodEntry;      public        Count: Word;        { $FFFF if there is no further info, or the value of Count }        RTTICount: Word;        { Entry: array[0..Count - 1] of TIntfMethodEntry }        property Method[Index: Word]: PIntfMethodEntry read GetMethod;      end;      PVmtMethodEntry = ^TVmtMethodEntry;      TVmtMethodEntry =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record        Name: PShortString;        CodeAddress: CodePointer;      end;      PVmtMethodTable = ^TVmtMethodTable;      TVmtMethodTable =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetEntry(Index: LongWord): PVmtMethodEntry; inline;      public        Count: LongWord;        property Entry[Index: LongWord]: PVmtMethodEntry read GetEntry;      private        Entries: array[0..0] of TVmtMethodEntry;      end;      TRecOpOffsetEntry =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record        ManagementOperator: CodePointer;        FieldOffset: SizeUInt;      end;      TRecOpOffsetTable =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record        Count: LongWord;        Entries: array[0..0] of TRecOpOffsetEntry;      end;      PRecOpOffsetTable = ^TRecOpOffsetTable;      PRecInitData = ^TRecInitData;      TRecInitData =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record        {$ifdef PROVIDE_ATTR_TABLE}        AttributeTable : PAttributeTable;        {$endif}        case TTypeKind of          tkRecord: (            Terminator: Pointer;            Size: Longint;{$ifndef VER3_0}            InitOffsetOp: PRecOpOffsetTable;            ManagementOp: Pointer;{$endif}            ManagedFieldCount: Longint;          { ManagedFields: array[0..ManagedFieldCount - 1] of TInitManagedField ; }          );          { include for proper alignment }          tkInt64: (            dummy : Int64          );      end;      PInterfaceData = ^TInterfaceData;      TInterfaceData =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetUnitName: ShortString; inline;        function GetPropertyTable: PPropData; inline;        function GetMethodTable: PIntfMethodTable; inline;      public        property UnitName: ShortString read GetUnitName;        property PropertyTable: PPropData read GetPropertyTable;        property MethodTable: PIntfMethodTable read GetMethodTable;      public      {$ifdef PROVIDE_ATTR_TABLE}        AttributeTable : PAttributeTable;      {$endif}      case TTypeKind of        tkInterface: (          Parent: PPTypeInfo;          Flags: TIntfFlagsBase;          GUID: TGUID;          UnitNameField: ShortString;          { PropertyTable: TPropData }          { MethodTable: TIntfMethodTable }        );        { include for proper alignment }        tkInt64: (          dummy : Int64        );{$ifndef FPUNONE}        tkFloat:          (FloatType : TFloatType        );{$endif}      end;      PInterfaceRawData = ^TInterfaceRawData;      TInterfaceRawData =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetUnitName: ShortString; inline;        function GetIIDStr: ShortString; inline;        function GetPropertyTable: PPropData; inline;        function GetMethodTable: PIntfMethodTable; inline;      public        property UnitName: ShortString read GetUnitName;        property IIDStr: ShortString read GetIIDStr;        property PropertyTable: PPropData read GetPropertyTable;        property MethodTable: PIntfMethodTable read GetMethodTable;      public      {$ifdef PROVIDE_ATTR_TABLE}        AttributeTable : PAttributeTable;      {$endif}        case TTypeKind of          tkInterface: (            Parent: PPTypeInfo;            Flags : TIntfFlagsBase;            IID: TGUID;            UnitNameField: ShortString;            { IIDStr: ShortString; }            { PropertyTable: TPropData }          );          { include for proper alignment }          tkInt64: (            dummy : Int64          );{$ifndef FPUNONE}          tkFloat:            (FloatType : TFloatType          );{$endif}      end;      PClassData = ^TClassData;      TClassData =      {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed      {$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetUnitName: ShortString; inline;        function GetPropertyTable: PPropData; inline;      public        property UnitName: ShortString read GetUnitName;        property PropertyTable: PPropData read GetPropertyTable;      public        {$ifdef PROVIDE_ATTR_TABLE}        AttributeTable : PAttributeTable;        {$endif}        case TTypeKind of          tkClass: (            ClassType : TClass;            Parent : PPTypeInfo;            PropCount : SmallInt;            UnitNameField : ShortString;            { PropertyTable: TPropData }          );          { include for proper alignment }          tkInt64: (            dummy: Int64;          );{$ifndef FPUNONE}          tkFloat: (            FloatType : TFloatType          );{$endif}      end;      PTypeData = ^TTypeData;      TTypeData ={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed{$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetBaseType: PTypeInfo; inline;        function GetCompType: PTypeInfo; inline;        function GetParentInfo: PTypeInfo; inline;{$ifndef VER3_0}                function GetRecInitData: PRecInitData; inline;{$endif}        function GetHelperParent: PTypeInfo; inline;        function GetExtendedInfo: PTypeInfo; inline;        function GetIntfParent: PTypeInfo; inline;        function GetRawIntfParent: PTypeInfo; inline;        function GetIIDStr: ShortString; inline;        function GetElType: PTypeInfo; inline;        function GetElType2: PTypeInfo; inline;        function GetInstanceType: PTypeInfo; inline;        function GetRefType: PTypeInfo; inline;      public        { tkEnumeration }        property BaseType: PTypeInfo read GetBaseType;        { tkSet }        property CompType: PTypeInfo read GetCompType;        { tkClass }        property ParentInfo: PTypeInfo read GetParentInfo;        { tkRecord }{$ifndef VER3_0}                property RecInitData: PRecInitData read GetRecInitData;{$endif}        { tkHelper }        property HelperParent: PTypeInfo read GetHelperParent;        property ExtendedInfo: PTypeInfo read GetExtendedInfo;        { tkInterface }        property IntfParent: PTypeInfo read GetIntfParent;        { tkInterfaceRaw }        property RawIntfParent: PTypeInfo read GetRawIntfParent;        property IIDStr: ShortString read GetIIDStr;        { tkDynArray }        property ElType2: PTypeInfo read GetElType2;        property ElType: PTypeInfo read GetElType;        { tkClassRef }        property InstanceType: PTypeInfo read GetInstanceType;        { tkPointer }        property RefType: PTypeInfo read GetRefType;      public         {$ifdef PROVIDE_ATTR_TABLE}         AttributeTable : PAttributeTable;         {$endif}         case TTypeKind of            tkUnKnown,tkLString,tkWString,tkVariant,tkUString:              ();            tkAString:              (CodePage: Word);{$ifndef VER3_0}            tkInt64,tkQWord,{$endif VER3_0}            tkInteger,tkChar,tkEnumeration,tkBool,tkWChar,tkSet:              (OrdType : TOrdType;               case TTypeKind of                  tkInteger,tkChar,tkEnumeration,tkBool,tkWChar : (                    MinValue,MaxValue : Longint;                    case TTypeKind of                      tkEnumeration:                        (                        BaseTypeRef : TypeInfoPtr;                        NameList : ShortString;                        {EnumUnitName: ShortString;})                    );{$ifndef VER3_0}                  {tkBool with OrdType=otSQWord }                  tkInt64:                    (MinInt64Value, MaxInt64Value: Int64);                  {tkBool with OrdType=otUQWord }                  tkQWord:                    (MinQWordValue, MaxQWordValue: QWord);{$endif VER3_0}                  tkSet:                    ({$ifndef VER3_0}                     SetSize : SizeInt;{$endif VER3_0}                     CompTypeRef : TypeInfoPtr                    )              );{$ifndef FPUNONE}            tkFloat:              (FloatType : TFloatType);{$endif}            tkSString:              (MaxLength : Byte);            tkClass:              (ClassType : TClass;               ParentInfoRef : TypeInfoPtr;               PropCount : SmallInt;               UnitName : ShortString;               // here the properties follow as array of TPropInfo              );            tkRecord:              ({$ifndef VER3_0}                RecInitInfo: Pointer; { points to TTypeInfo followed by init table }{$endif VER3_0}                RecSize: Longint;                case Boolean of                  False: (ManagedFldCount: Longint deprecated 'Use RecInitData^.ManagedFieldCount or TotalFieldCount depending on your use case');                  True: (TotalFieldCount: Longint);                {ManagedFields: array[1..TotalFieldCount] of TManagedField}              );            tkHelper:              (HelperParentRef : TypeInfoPtr;               ExtendedInfoRef : TypeInfoPtr;               HelperProps : SmallInt;               HelperUnit : ShortString               // here the properties follow as array of TPropInfo              );            tkMethod:              (MethodKind : TMethodKind;               ParamCount : Byte;               case Boolean of                 False: (ParamList : array[0..1023] of AnsiChar);                 { dummy for proper alignment }                 True: (ParamListDummy : Word);             {in reality ParamList is a array[1..ParamCount] of:                  record                    Flags : TParamFlags;                    ParamName : ShortString;                    TypeName : ShortString;                  end;              followed by                  ResultType : ShortString     // for mkFunction, mkClassFunction only                  ResultTypeRef : PPTypeInfo;  // for mkFunction, mkClassFunction only                  CC : TCallConv;                  ParamTypeRefs : array[1..ParamCount] of PPTypeInfo;}              );            tkProcVar:              (ProcSig: TProcedureSignature);{$ifdef VER3_0}            tkInt64:              (MinInt64Value, MaxInt64Value: Int64);            tkQWord:              (MinQWordValue, MaxQWordValue: QWord);{$endif VER3_0}            tkInterface:              (               IntfParentRef: TypeInfoPtr;               IntfFlags : TIntfFlagsBase;               GUID: TGUID;               IntfUnit: ShortString;               { PropertyTable: TPropData }               { MethodTable: TIntfMethodTable }              );            tkInterfaceRaw:              (               RawIntfParentRef: TypeInfoPtr;               RawIntfFlags : TIntfFlagsBase;               IID: TGUID;               RawIntfUnit: ShortString;               { IIDStr: ShortString; }               { PropertyTable: TPropData }              );            tkArray:              (ArrayData: TArrayTypeData);            tkDynArray:              (              elSize     : PtrUInt;              elType2Ref : TypeInfoPtr;              varType    : Longint;              elTypeRef  : TypeInfoPtr;              DynUnitName: ShortStringBase              );            tkClassRef:              (InstanceTypeRef: TypeInfoPtr);            tkPointer:              (RefTypeRef: TypeInfoPtr);      end;      PPropInfo = ^TPropInfo;      TPropData ={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}      packed{$endif FPC_REQUIRES_PROPER_ALIGNMENT}      record      private        function GetProp(Index: Word): PPropInfo;        function GetTail: Pointer; inline;      public        PropCount : Word;        PropList : record _alignmentdummy : ptrint; end;        property Prop[Index: Word]: PPropInfo read GetProp;        property Tail: Pointer read GetTail;      end;{$PACKRECORDS 1}      TPropInfo = packed record      private        function GetPropType: PTypeInfo; inline;        function GetTail: Pointer; inline;        function GetNext: PPropInfo; inline;      public        PropTypeRef : TypeInfoPtr;        GetProc : CodePointer;        SetProc : CodePointer;        StoredProc : CodePointer;        Index : Longint;        Default : Longint;        NameIndex : SmallInt;        // contains the type of the Get/Set/Storedproc, see also ptxxx        // bit 0..1 GetProc        //     2..3 SetProc        //     4..5 StoredProc        //     6 : true, constant index property        PropProcs : Byte;        {$ifdef PROVIDE_ATTR_TABLE}        AttributeTable : PAttributeTable;        {$endif}        Name : ShortString;        property PropType: PTypeInfo read GetPropType;        property Tail: Pointer read GetTail;        property Next: PPropInfo read GetNext;      end;      TProcInfoProc = Procedure(PropInfo : PPropInfo) of object;      PPropList = ^TPropList;      TPropList = array[0..{$ifdef cpu16}(32768 div sizeof(PPropInfo))-2{$else}65535{$endif}] of PPropInfo;   const      tkString = tkSString;      tkProcedure = tkProcVar; // for compatibility with Delphi      tkAny = [Low(TTypeKind)..High(TTypeKind)];      tkMethods = [tkMethod];      tkProperties = tkAny-tkMethods-[tkUnknown];// general property handlingFunction GetTypeData(TypeInfo : PTypeInfo) : PTypeData;Function AlignTypeData(p : Pointer) : Pointer; inline;Function AlignTParamFlags(p : Pointer) : Pointer; inline;Function AlignPTypeInfo(p : Pointer) : Pointer; inline;Generic Function ConstParamIsRef<T>(aCallConv: TCallConv = ccReg): Boolean; inline;Function GetPropInfo(TypeInfo: PTypeInfo;const PropName: string): PPropInfo;Function GetPropInfo(TypeInfo: PTypeInfo;const PropName: string; AKinds: TTypeKinds): PPropInfo;Function GetPropInfo(Instance: TObject; const PropName: string): PPropInfo;Function GetPropInfo(Instance: TObject; const PropName: string; AKinds: TTypeKinds): PPropInfo;Function GetPropInfo(AClass: TClass; const PropName: string): PPropInfo;Function GetPropInfo(AClass: TClass; const PropName: string; AKinds: TTypeKinds): PPropInfo;Function FindPropInfo(Instance: TObject; const PropName: string): PPropInfo;Function FindPropInfo(Instance: TObject; const PropName: string; AKinds: TTypeKinds): PPropInfo;Function FindPropInfo(AClass: TClass; const PropName: string): PPropInfo;Function FindPropInfo(AClass: TClass; const PropName: string; AKinds: TTypeKinds): PPropInfo;Procedure GetPropInfos(TypeInfo: PTypeInfo; PropList: PPropList);Function GetPropList(TypeInfo: PTypeInfo; TypeKinds: TTypeKinds; PropList: PPropList; Sorted: boolean = true): longint;Function GetPropList(TypeInfo: PTypeInfo; out PropList: PPropList): SizeInt;function GetPropList(AClass: TClass; out PropList: PPropList): Integer;function GetPropList(Instance: TObject; out PropList: PPropList): Integer;// Property information routines.Function IsReadableProp(PropInfo : PPropInfo) : Boolean;Function IsReadableProp(Instance: TObject; const PropName: string): Boolean;Function IsReadableProp(AClass: TClass; const PropName: string): Boolean;Function IsWriteableProp(PropInfo : PPropInfo) : Boolean;Function IsWriteableProp(Instance: TObject; const PropName: string): Boolean;Function IsWriteableProp(AClass: TClass; const PropName: string): Boolean;Function IsStoredProp(Instance: TObject;PropInfo : PPropInfo) : Boolean;Function IsStoredProp(Instance: TObject; const PropName: string): Boolean;Function IsPublishedProp(Instance: TObject; const PropName: string): Boolean;Function IsPublishedProp(AClass: TClass; const PropName: string): Boolean;Function PropType(Instance: TObject; const PropName: string): TTypeKind;Function PropType(AClass: TClass; const PropName: string): TTypeKind;Function PropIsType(Instance: TObject; const PropName: string; TypeKind: TTypeKind): Boolean;Function PropIsType(AClass: TClass; const PropName: string; TypeKind: TTypeKind): Boolean;// subroutines to read/write propertiesFunction  GetOrdProp(Instance: TObject; PropInfo : PPropInfo) : Int64;Function  GetOrdProp(Instance: TObject; const PropName: string): Int64;Procedure SetOrdProp(Instance: TObject; PropInfo : PPropInfo;  Value : Int64);Procedure SetOrdProp(Instance: TObject; const PropName: string; Value: Int64);Function  GetEnumProp(Instance: TObject; const PropName: string): string;Function  GetEnumProp(Instance: TObject; const PropInfo: PPropInfo): string;Procedure SetEnumProp(Instance: TObject; const PropName: string;const Value: string);Procedure SetEnumProp(Instance: TObject; const PropInfo: PPropInfo;const Value: string);Function  GetSetProp(Instance: TObject; const PropName: string): string;Function  GetSetProp(Instance: TObject; const PropName: string; Brackets: Boolean): string;Function  GetSetProp(Instance: TObject; const PropInfo: PPropInfo; Brackets: Boolean): string;Procedure SetSetProp(Instance: TObject; const PropName: string; const Value: string);Procedure SetSetProp(Instance: TObject; const PropInfo: PPropInfo; const Value: string);Function  GetStrProp(Instance: TObject; PropInfo : PPropInfo) : Ansistring;Function  GetStrProp(Instance: TObject; const PropName: string): string;Procedure SetStrProp(Instance: TObject; const PropName: string; const Value: AnsiString);Procedure SetStrProp(Instance: TObject; PropInfo : PPropInfo; const Value : Ansistring);Function GetWideStrProp(Instance: TObject; PropInfo: PPropInfo): WideString;Function GetWideStrProp(Instance: TObject; const PropName: string): WideString;Procedure SetWideStrProp(Instance: TObject; const PropName: string; const Value: WideString);Procedure SetWideStrProp(Instance: TObject; PropInfo: PPropInfo; const Value: WideString);Function GetUnicodeStrProp(Instance: TObject; PropInfo: PPropInfo): UnicodeString;Function GetUnicodeStrProp(Instance: TObject; const PropName: string): UnicodeString;Procedure SetUnicodeStrProp(Instance: TObject; const PropName: string; const Value: UnicodeString);Procedure SetUnicodeStrProp(Instance: TObject; PropInfo: PPropInfo; const Value: UnicodeString);Function GetRawbyteStrProp(Instance: TObject; PropInfo: PPropInfo): RawByteString;Function GetRawByteStrProp(Instance: TObject; const PropName: string): RawByteString;Procedure SetRawByteStrProp(Instance: TObject; const PropName: string; const Value: RawByteString);Procedure SetRawByteStrProp(Instance: TObject; PropInfo: PPropInfo; const Value: RawByteString);{$ifndef FPUNONE}Function  GetFloatProp(Instance: TObject; PropInfo : PPropInfo) : Extended;Function  GetFloatProp(Instance: TObject; const PropName: string): Extended;Procedure SetFloatProp(Instance: TObject; const PropName: string; Value: Extended);Procedure SetFloatProp(Instance: TObject; PropInfo : PPropInfo;  Value : Extended);{$endif}Function  GetObjectProp(Instance: TObject; const PropName: string): TObject;Function  GetObjectProp(Instance: TObject; const PropName: string; MinClass: TClass): TObject;Function  GetObjectProp(Instance: TObject; PropInfo: PPropInfo): TObject;Function  GetObjectProp(Instance: TObject; PropInfo: PPropInfo; MinClass: TClass): TObject;Procedure SetObjectProp(Instance: TObject; const PropName: string; Value: TObject);Procedure SetObjectProp(Instance: TObject; PropInfo: PPropInfo; Value: TObject);Function  GetObjectPropClass(Instance: TObject; const PropName: string): TClass;Function  GetObjectPropClass(AClass: TClass; const PropName: string): TClass;Function  GetMethodProp(Instance: TObject; PropInfo: PPropInfo) : TMethod;Function  GetMethodProp(Instance: TObject; const PropName: string): TMethod;Procedure SetMethodProp(Instance: TObject; PropInfo: PPropInfo;  const Value : TMethod);Procedure SetMethodProp(Instance: TObject; const PropName: string; const Value: TMethod);Function  GetInt64Prop(Instance: TObject; PropInfo: PPropInfo): Int64;Function  GetInt64Prop(Instance: TObject; const PropName: string): Int64;Procedure SetInt64Prop(Instance: TObject; PropInfo: PPropInfo;  const Value: Int64);Procedure SetInt64Prop(Instance: TObject; const PropName: string;  const Value: Int64);Function GetPropValue(Instance: TObject; const PropName: string): Variant;Function GetPropValue(Instance: TObject; const PropName: string; PreferStrings: Boolean): Variant;Function GetPropValue(Instance: TObject; PropInfo: PPropInfo): Variant;Function GetPropValue(Instance: TObject; PropInfo: PPropInfo; PreferStrings: Boolean): Variant;Procedure SetPropValue(Instance: TObject; const PropName: string; const Value: Variant);Procedure SetPropValue(Instance: TObject; PropInfo: PPropInfo; const Value: Variant);Function  GetVariantProp(Instance: TObject; PropInfo : PPropInfo): Variant;Function  GetVariantProp(Instance: TObject; const PropName: string): Variant;Procedure SetVariantProp(Instance: TObject; const PropName: string; const Value: Variant);Procedure SetVariantProp(Instance: TObject; PropInfo : PPropInfo; const Value: Variant);function GetInterfaceProp(Instance: TObject; const PropName: string): IInterface;function GetInterfaceProp(Instance: TObject; PropInfo: PPropInfo): IInterface;procedure SetInterfaceProp(Instance: TObject; const PropName: string; const Value: IInterface);procedure SetInterfaceProp(Instance: TObject; PropInfo: PPropInfo; const Value: IInterface);function GetRawInterfaceProp(Instance: TObject; const PropName: string): Pointer;function GetRawInterfaceProp(Instance: TObject; PropInfo: PPropInfo): Pointer;procedure SetRawInterfaceProp(Instance: TObject; const PropName: string; const Value: Pointer);procedure SetRawInterfaceProp(Instance: TObject; PropInfo: PPropInfo; const Value: Pointer);function GetDynArrayProp(Instance: TObject; const PropName: string): Pointer;function GetDynArrayProp(Instance: TObject; PropInfo: PPropInfo): Pointer;procedure SetDynArrayProp(Instance: TObject; const PropName: string; const Value: Pointer);procedure SetDynArrayProp(Instance: TObject; PropInfo: PPropInfo; const Value: Pointer);// Extended RTTIfunction GetAttributeTable(TypeInfo: PTypeInfo): PAttributeTable;function GetPropAttribute(PropInfo: PPropInfo; AttributeNr: Word): TCustomAttribute; inline;function GetAttribute(AttributeTable: PAttributeTable; AttributeNr: Word): TCustomAttribute;// Auxiliary routines, which may be usefulFunction GetEnumName(TypeInfo : PTypeInfo;Value : Integer) : string;Function GetEnumValue(TypeInfo : PTypeInfo;const Name : string) : Integer;function GetEnumNameCount(enum1: PTypeInfo): SizeInt;procedure AddEnumElementAliases(aTypeInfo: PTypeInfo; const aNames: array of string; aStartValue: Integer = 0);procedure RemoveEnumElementAliases(aTypeInfo: PTypeInfo);function GetEnumeratedAliasValue(aTypeInfo: PTypeInfo; const aName: string): Integer;function SetToArray(TypeInfo: PTypeInfo; Value: Pointer) : TBytes;function SetToArray(PropInfo: PPropInfo; Value: Pointer) : TBytes;function SetToArray(TypeInfo: PTypeInfo; Value: LongInt) : TBytes;function SetToArray(PropInfo: PPropInfo; Value: LongInt) : TBytes;function SetToString(TypeInfo: PTypeInfo; Value: LongInt; Brackets: Boolean) : String;function SetToString(PropInfo: PPropInfo; Value: LongInt; Brackets: Boolean) : String;function SetToString(PropInfo: PPropInfo; Value: LongInt) : String;function SetToString(TypeInfo: PTypeInfo; Value: Pointer; Brackets: Boolean = False) : String;function SetToString(PropInfo: PPropInfo; Value: Pointer; Brackets: Boolean = False) : String;function ArrayToSet(PropInfo: PPropInfo; const Value: array of Byte): LongInt;function ArrayToSet(TypeInfo: PTypeInfo; const Value: array of Byte): LongInt;procedure ArrayToSet(PropInfo: PPropInfo; const Value: array of Byte; Result: Pointer);procedure ArrayToSet(TypeInfo: PTypeInfo; const Value: array of Byte; Result: Pointer);function StringToSet(PropInfo: PPropInfo; const Value: string): LongInt;function StringToSet(TypeInfo: PTypeInfo; const Value: string): LongInt;procedure StringToSet(PropInfo: PPropInfo; const Value: String; Result: Pointer);procedure StringToSet(TypeInfo: PTypeInfo; const Value: String; Result: Pointer);const    BooleanIdents: array[Boolean] of String = ('False', 'True');    DotSep: String = '.';Type  EPropertyError  = Class(Exception);  TGetPropValue   = Function (Instance: TObject; PropInfo: PPropInfo; PreferStrings: Boolean) : Variant;  TSetPropValue   = Procedure (Instance: TObject; PropInfo: PPropInfo; const Value: Variant);  TGetVariantProp = Function (Instance: TObject; PropInfo : PPropInfo): Variant;  TSetVariantProp = Procedure (Instance: TObject; PropInfo : PPropInfo; const Value: Variant);  EPropertyConvertError = class(Exception); // Not used (yet), but defined for compatibility.Const  OnGetPropValue   : TGetPropValue = Nil;  OnSetPropValue   : TSetPropValue = Nil;  OnGetVariantprop : TGetVariantProp = Nil;  OnSetVariantprop : TSetVariantProp = Nil;{ for inlining }function DerefTypeInfoPtr(Info: TypeInfoPtr): PTypeInfo; inline;Implementationuses rtlconsts;type  PMethod = ^TMethod;{ ---------------------------------------------------------------------  Auxiliary methods  ---------------------------------------------------------------------}function aligntoptr(p : pointer) : pointer;inline;   begin{$ifdef CPUM68K}     result:=AlignTypeData(p);{$else CPUM68K}{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}     result:=align(p,sizeof(p));{$else FPC_REQUIRES_PROPER_ALIGNMENT}     result:=p;{$endif FPC_REQUIRES_PROPER_ALIGNMENT}{$endif CPUM68K}   end;function DerefTypeInfoPtr(Info: TypeInfoPtr): PTypeInfo; inline;begin{$ifdef ver3_0}  Result := Info;{$else}  if not Assigned(Info) then    Result := Nil  else    Result := Info^;{$endif}end;function GetAttributeTable(TypeInfo: PTypeInfo): PAttributeTable;{$ifdef PROVIDE_ATTR_TABLE}var  TD: PTypeData;begin  TD := GetTypeData(TypeInfo);  Result:=TD^.AttributeTable;{$else}begin  Result:=Nil;{$endif}end;function GetPropData(TypeInfo : PTypeInfo; TypeData: PTypeData) : PPropData; inline;var  p: PtrUInt;begin  p := PtrUInt(@TypeData^.UnitName) + SizeOf(TypeData^.UnitName[0]) + Length(TypeData^.UnitName);  Result := PPropData(aligntoptr(Pointer(p)));end;function GetAttribute(AttributeTable: PAttributeTable; AttributeNr: Word): TCustomAttribute;begin  if (AttributeTable=nil) or (AttributeNr>=AttributeTable^.AttributeCount) then    result := nil  else    begin      result := AttributeTable^.AttributesList[AttributeNr].AttrProc();    end;end;function GetPropAttribute(PropInfo: PPropInfo; AttributeNr: Word): TCustomAttribute;begin{$ifdef PROVIDE_ATTR_TABLE}  Result := GetAttribute(PropInfo^.AttributeTable, AttributeNr);{$else}  Result := Nil;{$endif}end;Function GetEnumName(TypeInfo : PTypeInfo;Value : Integer) : string;  Var PS : PShortString;      PT : PTypeData;begin  PT:=GetTypeData(TypeInfo);  if TypeInfo^.Kind=tkBool then    begin      case Value of        0,1:          Result:=BooleanIdents[Boolean(Value)];        else          Result:='';      end;    end else   begin     PS:=@PT^.NameList;     dec(Value,PT^.MinValue);     While Value>0 Do       begin         PS:=PShortString(pointer(PS)+PByte(PS)^+1);         Dec(Value);       end;     Result:=PS^;   end;end;Function GetEnumValue(TypeInfo : PTypeInfo;const Name : string) : Integer;  Var PS : PShortString;      PT : PTypeData;      Count : longint;      sName: shortstring;begin  If Length(Name)=0 then    exit(-1);  sName := Name;  PT:=GetTypeData(TypeInfo);  Count:=0;  Result:=-1;  if TypeInfo^.Kind=tkBool then    begin    If CompareText(BooleanIdents[false],Name)=0 then      result:=0    else if CompareText(BooleanIdents[true],Name)=0 then      result:=1;    end else   begin     PS:=@PT^.NameList;     While (Result=-1) and (PByte(PS)^<>0) do       begin         If ShortCompareText(PS^, sName) = 0 then           Result:=Count+PT^.MinValue;         PS:=PShortString(pointer(PS)+PByte(PS)^+1);         Inc(Count);       end;     if Result=-1 then       Result:=GetEnumeratedAliasValue(TypeInfo,Name);   end;end;function GetEnumNameCount(enum1: PTypeInfo): SizeInt;var  PS: PShortString;  PT: PTypeData;  Count: SizeInt;begin  PT:=GetTypeData(enum1);  if enum1^.Kind=tkBool then    Result:=2  else    begin      Count:=0;      Result:=0;      PS:=@PT^.NameList;      While (PByte(PS)^<>0) do        begin          PS:=PShortString(pointer(PS)+PByte(PS)^+1);          Inc(Count);        end;      { the last string is the unit name }      Result := Count - 1;    end;end;Function SetToString(PropInfo: PPropInfo; Value: LongInt; Brackets: Boolean) : String;begin  Result:=SetToString(PropInfo^.PropType, Value, Brackets);end;Function SetToString(TypeInfo: PTypeInfo; Value: LongInt; Brackets: Boolean) : String;begin{$if defined(FPC_BIG_ENDIAN)}  { correctly adjust packed sets that are smaller than 32-bit }  case GetTypeData(TypeInfo)^.OrdType of    otSByte,otUByte: Value := Value shl (SizeOf(Integer)*8-8);    otSWord,otUWord: Value := Value shl (SizeOf(Integer)*8-16);  end;{$endif}  Result := SetToString(TypeInfo, @Value, Brackets);end;function SetToString(TypeInfo: PTypeInfo; Value: Pointer; Brackets: Boolean): String;var  A: TBytes;  B: Byte;  PTI : PTypeInfo;begin  PTI:=GetTypeData(TypeInfo)^.CompType;  A:=SetToArray(TypeInfo, Value);  Result := '';  for B in A do    If Result='' then      Result:=GetEnumName(PTI,B)    else      Result:=Result+','+GetEnumName(PTI,B);  if Brackets then    Result:='['+Result+']';end;Function SetToString(PropInfo: PPropInfo; Value: LongInt) : String;begin  Result:=SetToString(PropInfo,Value,False);end;function SetToString(PropInfo: PPropInfo; Value: Pointer; Brackets: Boolean): String;begin  Result := SetToString(PropInfo^.PropType, Value, Brackets);end;function SetToArray(TypeInfo: PTypeInfo; Value: Pointer) : TBytes;type  tsetarr = bitpacked array[0..SizeOf(LongInt)*8-1] of 0..1;Var  I,El,Els,Rem,V,Max : Integer;  PTD : PTypeData;  ValueArr : PLongInt;begin  PTD := GetTypeData(TypeInfo);  ValueArr := PLongInt(Value);  Result:=[];{$ifdef ver3_0}  case PTD^.OrdType of    otSByte, otUByte: begin      Els := 0;      Rem := 1;    end;    otSWord, otUWord: begin      Els := 0;      Rem := 2;    end;    otSLong, otULong: begin      Els := 1;      Rem := 0;    end;  end;{$else}  Els := PTD^.SetSize div SizeOf(LongInt);  Rem := PTD^.SetSize mod SizeOf(LongInt);{$endif}{$ifdef ver3_0}  El := 0;{$else}  for El := 0 to (PTD^.SetSize - 1) div SizeOf(LongInt) do{$endif}    begin      if El = Els then        Max := Rem      else        Max := SizeOf(LongInt);      For I:=0 to Max*8-1 do        begin          if (tsetarr(ValueArr[El])[i]<>0) then            begin              V := I + SizeOf(LongInt) * 8 * El;              SetLength(Result, Length(Result)+1);              Result[High(Result)]:=V;            end;        end;    end;end;function SetToArray(PropInfo: PPropInfo; Value: Pointer) : TBytes;begin  Result:=SetToArray(PropInfo^.PropType,Value);end;function SetToArray(TypeInfo: PTypeInfo; Value: LongInt) : TBytes;begin  Result:=SetToArray(TypeInfo,@Value);end;function SetToArray(PropInfo: PPropInfo; Value: LongInt) : TBytes;begin  Result:=SetToArray(PropInfo^.PropType,@Value);end;Const  SetDelim = ['[',']',',',' '];Function GetNextElement(Var S : String) : String;Var  J : Integer;begin  J:=1;  Result:='';  If Length(S)>0 then    begin      While (J<=Length(S)) and Not (S[j] in SetDelim) do        Inc(j);      Result:=Copy(S,1,j-1);      Delete(S,1,j);    end;end;Function StringToSet(PropInfo: PPropInfo; const Value: string): LongInt;begin  Result:=StringToSet(PropInfo^.PropType,Value);end;Function StringToSet(TypeInfo: PTypeInfo; const Value: string): LongInt;begin  StringToSet(TypeInfo, Value, @Result);{$if defined(FPC_BIG_ENDIAN)}  { correctly adjust packed sets that are smaller than 32-bit }  case GetTypeData(TypeInfo)^.OrdType of    otSByte,otUByte: Result := Result shr (SizeOf(Integer)*8-8);    otSWord,otUWord: Result := Result shr (SizeOf(Integer)*8-16);  end;{$endif}end;procedure StringToSet(TypeInfo: PTypeInfo; const Value: String; Result: Pointer);Var  S,T : String;  I, ElOfs, BitOfs : Integer;  PTD: PTypeData;  PTI : PTypeInfo;  A: TBytes;begin  PTD:=GetTypeData(TypeInfo);  PTI:=PTD^.Comptype;  S:=Value;  I:=1;  If Length(S)>0 then    begin      While (I<=Length(S)) and (S[i] in SetDelim) do        Inc(I);      Delete(S,1,i-1);    end;  A:=[];  While (S<>'') do    begin      T:=GetNextElement(S);      if T<>'' then        begin          I:=GetEnumValue(PTI,T);          if (I<0) then            raise EPropertyError.CreateFmt(SErrUnknownEnumValue, [T]);          SetLength(A, Length(A)+1);          A[High(A)]:=I;        end;    end;  ArrayToSet(TypeInfo,A,Result);end;procedure StringToSet(PropInfo: PPropInfo; const Value: String; Result: Pointer);begin  StringToSet(PropInfo^.PropType, Value, Result);end;Function ArrayToSet(PropInfo: PPropInfo; const Value: array of Byte): LongInt;begin  Result:=ArrayToSet(PropInfo^.PropType,Value);end;Function ArrayToSet(TypeInfo: PTypeInfo; const Value: array of Byte): LongInt;begin  ArrayToSet(TypeInfo, Value, @Result);{$if defined(FPC_BIG_ENDIAN)}  { correctly adjust packed sets that are smaller than 32-bit }  case GetTypeData(TypeInfo)^.OrdType of    otSByte,otUByte: Result := Result shr (SizeOf(Integer)*8-8);    otSWord,otUWord: Result := Result shr (SizeOf(Integer)*8-16);  end;{$endif}end;procedure ArrayToSet(TypeInfo: PTypeInfo; const Value: array of Byte; Result: Pointer);Var  ElOfs, BitOfs : Integer;  PTD: PTypeData;  ResArr: PLongWord;  B: Byte;begin  PTD:=GetTypeData(TypeInfo);{$ifndef ver3_0}  FillChar(Result^, PTD^.SetSize, 0);{$else}  PInteger(Result)^ := 0;{$endif}  ResArr := PLongWord(Result);  for B in Value do    begin      ElOfs := B shr 5;      BitOfs := B and $1F;{$ifdef FPC_BIG_ENDIAN}      { on Big Endian systems enum values start from the MSB, thus we need        to reverse the shift }      BitOfs := 31 - BitOfs;{$endif}      ResArr[ElOfs] := ResArr[ElOfs] or (LongInt(1) shl BitOfs);    end;end;procedure ArrayToSet(PropInfo: PPropInfo; const Value: array of Byte; Result: Pointer);begin  ArrayToSet(PropInfo^.PropType, Value, Result);end;Function AlignTypeData(p : Pointer) : Pointer;{$packrecords c}  type    TAlignCheck = record      b : byte;      q : qword;    end;{$packrecords default}begin{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}{$ifdef VER3_0}  Result:=Pointer(align(p,SizeOf(Pointer)));{$else VER3_0}  Result:=Pointer(align(p,PtrInt(@TAlignCheck(nil^).q))){$endif VER3_0}{$else FPC_REQUIRES_PROPER_ALIGNMENT}  Result:=p;{$endif FPC_REQUIRES_PROPER_ALIGNMENT}end;Function AlignTParamFlags(p : Pointer) : Pointer; inline;{$packrecords c}  type    TAlignCheck = record      b : byte;      w : word;    end;{$packrecords default}begin{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}  Result:=Pointer(align(p,PtrInt(@TAlignCheck(nil^).w))){$else FPC_REQUIRES_PROPER_ALIGNMENT}  Result:=p;{$endif FPC_REQUIRES_PROPER_ALIGNMENT}end;Function AlignPTypeInfo(p : Pointer) : Pointer; inline;{$packrecords c}  type    TAlignCheck = record      b : byte;      p : pointer;    end;{$packrecords default}begin{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}  Result:=Pointer(align(p,PtrInt(@TAlignCheck(nil^).p))){$else FPC_REQUIRES_PROPER_ALIGNMENT}  Result:=p;{$endif FPC_REQUIRES_PROPER_ALIGNMENT}end;Generic Function ConstParamIsRef<T>(aCallConv: TCallConv): Boolean;  Function SameAddrRegister(const aArg1: T; constref aArg2: T): Boolean; register;  begin    Result := @aArg1 = @aArg2;  end;  Function SameAddrCDecl(const aArg1: T; constref aArg2: T): Boolean; cdecl;  begin    Result := @aArg1 = @aArg2;  end;{$if defined(cpui8086) or defined(cpui386)}  Function SameAddrPascal(const aArg1: T; constref aArg2: T): Boolean; pascal;  begin    Result := @aArg1 = @aArg2;  end;{$endif}  Function SameAddrStdCall(const aArg1: T; constref aArg2: T): Boolean; stdcall;  begin    Result := @aArg1 = @aArg2;  end;  Function SameAddrCppDecl(const aArg1: T; constref aArg2: T): Boolean; cppdecl;  begin    Result := @aArg1 = @aArg2;  end;{$if defined(cpui386)}  Function SameAddrOldFPCCall(const aArg1: T; constref aArg2: T): Boolean; oldfpccall;  begin    Result := @aArg1 = @aArg2;  end;{$endif}  Function SameAddrMWPascal(const aArg1: T; constref aArg2: T): Boolean; mwpascal;  begin    Result := @aArg1 = @aArg2;  end;var  v: T;begin  v := Default(T);  case aCallConv of    ccReg:      Result := SameAddrRegister(v, v);    ccCdecl:      Result := SameAddrCDecl(v, v);{$if defined(cpui386) or defined(cpui8086)}    ccPascal:      Result := SameAddrPascal(v, v);{$endif}{$if not defined(cpui386)}    ccOldFPCCall,{$endif}{$if not defined(cpui386) and not defined(cpui8086)}    ccPascal,{$endif}    ccStdCall:      Result := SameAddrStdCall(v, v);    ccCppdecl:      Result := SameAddrCppDecl(v, v);{$if defined(cpui386)}    ccOldFPCCall:      Result := SameAddrOldFPCCall(v, v);{$endif}    ccMWPascal:      Result := SameAddrMWPascal(v, v);    else      raise EArgumentException.CreateFmt(SUnsupportedCallConv, [GetEnumName(PTypeInfo(TypeInfo(TCallConv)), Ord(aCallConv))]);  end;end;Function GetTypeData(TypeInfo : PTypeInfo) : PTypeData;begin  GetTypeData:=AlignTypeData(pointer(TypeInfo)+2+PByte(pointer(TypeInfo)+1)^);end;{ ---------------------------------------------------------------------  Basic Type information functions.  ---------------------------------------------------------------------}Function GetPropInfo(TypeInfo : PTypeInfo;const PropName : string) : PPropInfo;var  hp : PTypeData;  i : longint;  p : shortstring;  pd : PPropData;begin  P:=PropName;  // avoid Ansi<->short conversion in a loop  while Assigned(TypeInfo) do    begin      // skip the name      hp:=GetTypeData(Typeinfo);      // the class info rtti the property rtti follows immediatly      pd := GetPropData(TypeInfo,hp);      Result:=PPropInfo(@pd^.PropList);      for i:=1 to pd^.PropCount do        begin          // found a property of that name ?          if ShortCompareText(Result^.Name, P) = 0 then            exit;          // skip to next property          Result:=PPropInfo(aligntoptr(pointer(@Result^.Name)+byte(Result^.Name[0])+1));        end;      // parent class      Typeinfo:=hp^.ParentInfo;    end;  Result:=Nil;end;Function GetPropInfo(TypeInfo : PTypeInfo;const PropName : string; Akinds : TTypeKinds) : PPropInfo;begin  Result:=GetPropInfo(TypeInfo,PropName);  If (Akinds<>[]) then    If (Result<>Nil) then      If Not (Result^.PropType^.Kind in AKinds) then        Result:=Nil;end;Function GetPropInfo(AClass: TClass; const PropName: string; AKinds: TTypeKinds) : PPropInfo;begin  Result:=GetPropInfo(PTypeInfo(AClass.ClassInfo),PropName,AKinds);end;Function GetPropInfo(Instance: TObject; const PropName: string; AKinds: TTypeKinds) : PPropInfo;begin  Result:=GetPropInfo(Instance.ClassType,PropName,AKinds);end;Function GetPropInfo(Instance: TObject; const PropName: string): PPropInfo;begin  Result:=GetPropInfo(Instance,PropName,[]);end;Function GetPropInfo(AClass: TClass; const PropName: string): PPropInfo;begin  Result:=GetPropInfo(AClass,PropName,[]);end;Function FindPropInfo(Instance: TObject; const PropName: string): PPropInfo;begin  result:=GetPropInfo(Instance, PropName);  if Result=nil then    Raise EPropertyError.CreateFmt(SErrPropertyNotFound, [PropName]);end;Function FindPropInfo(Instance: TObject; const PropName: string; AKinds: TTypeKinds): PPropInfo;begin  result:=GetPropInfo(Instance, PropName, AKinds);  if Result=nil then    Raise EPropertyError.CreateFmt(SErrPropertyNotFound, [PropName]);end;Function FindPropInfo(AClass: TClass; const PropName: string): PPropInfo;begin  result:=GetPropInfo(AClass, PropName);  if result=nil then    Raise EPropertyError.CreateFmt(SErrPropertyNotFound, [PropName]);end;Function FindPropInfo(AClass: TClass; const PropName: string; AKinds: TTypeKinds): PPropInfo;begin  result:=GetPropInfo(AClass, PropName, AKinds);  if result=nil then    Raise EPropertyError.CreateFmt(SErrPropertyNotFound, [PropName]);end;function IsReadableProp(PropInfo: PPropInfo): Boolean;begin  Result:=(((PropInfo^.PropProcs) and 3) in [ptField,ptStatic,ptVirtual]);end;function IsReadableProp(Instance: TObject; const PropName: string): Boolean;begin  Result:=IsReadableProp(FindPropInfo(Instance,PropName));end;function IsReadableProp(AClass: TClass; const PropName: string): Boolean;begin  Result:=IsReadableProp(FindPropInfo(AClass,PropName));end;function IsWriteableProp(PropInfo: PPropInfo): Boolean;begin  Result:=(((PropInfo^.PropProcs shr 2) and 3) in [ptField,ptStatic,ptVirtual]);end;function IsWriteableProp(Instance: TObject; const PropName: string): Boolean;begin  Result:=IsWriteableProp(FindPropInfo(Instance,PropName));end;function IsWriteableProp(AClass: TClass; const PropName: string): Boolean;begin  Result:=IsWriteableProp(FindPropInfo(AClass,PropName));end;Function IsStoredProp(Instance : TObject;PropInfo : PPropInfo) : Boolean;type  TBooleanIndexFunc=function(Index:integer):boolean of object;  TBooleanFunc=function:boolean of object;var  AMethod : TMethod;begin  case (PropInfo^.PropProcs shr 4) and 3 of    ptField:      Result:=PBoolean(Pointer(Instance)+PtrUInt(PropInfo^.StoredProc))^;    ptConst:      Result:=LongBool(PropInfo^.StoredProc);    ptStatic,    ptVirtual:      begin        if (PropInfo^.PropProcs shr 4) and 3=ptstatic then          AMethod.Code:=PropInfo^.StoredProc        else          AMethod.Code:=pcodepointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.StoredProc))^;        AMethod.Data:=Instance;        if ((PropInfo^.PropProcs shr 6) and 1)<>0 then           Result:=TBooleanIndexFunc(AMethod)(PropInfo^.Index)        else           Result:=TBooleanFunc(AMethod)();      end;  end;end;Procedure GetPropInfos(TypeInfo : PTypeInfo;PropList : PPropList);{        Store Pointers to property information in the list pointed        to by proplist. PRopList must contain enough space to hold ALL        properties.}Var  TD : PTypeData;  TP : PPropInfo;  Count : Longint;begin  // Get this objects TOTAL published properties count  TD:=GetTypeData(TypeInfo);  // Clear list  FillChar(PropList^,TD^.PropCount*sizeof(Pointer),0);  repeat    TD:=GetTypeData(TypeInfo);    // published properties count for this object    TP:=PPropInfo(GetPropData(TypeInfo, TD));    Count:=PWord(TP)^;    // Now point TP to first propinfo record.    Inc(Pointer(TP),SizeOF(Word));    tp:=aligntoptr(tp);    While Count>0 do      begin        // Don't overwrite properties with the same name        if PropList^[TP^.NameIndex]=nil then          PropList^[TP^.NameIndex]:=TP;        // Point to TP next propinfo record.        // Located at Name[Length(Name)+1] !        TP:=aligntoptr(PPropInfo(pointer(@TP^.Name)+PByte(@TP^.Name)^+1));        Dec(Count);      end;    TypeInfo:=TD^.Parentinfo;  until TypeInfo=nil;end;Procedure InsertProp (PL : PProplist;PI : PPropInfo; Count : longint);Var  I : Longint;begin  I:=0;  While (I<Count) and (PI^.Name>PL^[I]^.Name) do    Inc(I);  If I<Count then    Move(PL^[I], PL^[I+1], (Count - I) * SizeOf(Pointer));  PL^[I]:=PI;end;Procedure InsertPropnosort (PL : PProplist;PI : PPropInfo; Count : longint);begin  PL^[Count]:=PI;end;Type TInsertProp = Procedure (PL : PProplist;PI : PPropInfo; Count : longint);//Const InsertProps : array[false..boolean] of TInsertProp = (InsertPropNoSort,InsertProp);Function  GetPropList(TypeInfo : PTypeInfo;TypeKinds : TTypeKinds; PropList : PPropList;Sorted : boolean = true):longint;{  Store Pointers to property information OF A CERTAIN KIND in the list pointed  to by proplist. PRopList must contain enough space to hold ALL  properties.}Var  TempList : PPropList;  PropInfo : PPropinfo;  I,Count : longint;  DoInsertProp : TInsertProp;begin  if sorted then    DoInsertProp:=@InsertProp  else    DoInsertProp:=@InsertPropnosort;  Result:=0;  Count:=GetTypeData(TypeInfo)^.Propcount;  If Count>0 then    begin      GetMem(TempList,Count*SizeOf(Pointer));      Try        GetPropInfos(TypeInfo,TempList);        For I:=0 to Count-1 do          begin            PropInfo:=TempList^[i];            If PropInfo^.PropType^.Kind in TypeKinds then              begin                If (PropList<>Nil) then                  DoInsertProp(PropList,PropInfo,Result);                Inc(Result);              end;          end;      finally        FreeMem(TempList,Count*SizeOf(Pointer));      end;    end;end;Function GetPropList(TypeInfo: PTypeInfo; out PropList: PPropList): SizeInt;begin  result:=GetTypeData(TypeInfo)^.Propcount;  if result>0 then    begin      getmem(PropList,result*sizeof(pointer));      GetPropInfos(TypeInfo,PropList);    end  else    PropList:=Nil;end;function GetPropList(AClass: TClass; out PropList: PPropList): Integer;begin  Result := GetPropList(PTypeInfo(AClass.ClassInfo), PropList);end;function GetPropList(Instance: TObject; out PropList: PPropList): Integer;begin  Result := GetPropList(Instance.ClassType, PropList);end;{ ---------------------------------------------------------------------  Property access functions  ---------------------------------------------------------------------}{ ---------------------------------------------------------------------  Ordinal properties  ---------------------------------------------------------------------}Function GetOrdProp(Instance : TObject;PropInfo : PPropInfo) : Int64;type  TGetInt64ProcIndex=function(index:longint):Int64 of object;  TGetInt64Proc=function():Int64 of object;  TGetIntegerProcIndex=function(index:longint):longint of object;  TGetIntegerProc=function:longint of object;  TGetWordProcIndex=function(index:longint):word of object;  TGetWordProc=function:word of object;  TGetByteProcIndex=function(index:longint):Byte of object;  TGetByteProc=function:Byte of object;var  TypeInfo: PTypeInfo;  AMethod : TMethod;  DataSize: Integer;  OrdType: TOrdType;  Signed: Boolean;begin  Result:=0;  TypeInfo := PropInfo^.PropType;  Signed := false;  DataSize := 4;  case TypeInfo^.Kind of// We keep this for backwards compatibility, but internally it is no longer used.{$ifdef cpu64}    tkInterface,    tkInterfaceRaw,    tkDynArray,    tkClass:      DataSize:=8;{$endif cpu64}    tkChar, tkBool:      DataSize:=1;    tkWChar:      DataSize:=2;    tkSet,    tkEnumeration,    tkInteger:      begin        OrdType:=GetTypeData(TypeInfo)^.OrdType;        case OrdType of          otSByte,otUByte: DataSize := 1;          otSWord,otUWord: DataSize := 2;        end;        Signed := OrdType in [otSByte,otSWord,otSLong];      end;    tkInt64 :      begin        DataSize:=8;        Signed:=true;      end;    tkQword :      begin        DataSize:=8;        Signed:=false;      end;  end;  case (PropInfo^.PropProcs) and 3 of    ptField:      if Signed then begin        case DataSize of          1: Result:=PShortInt(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;          2: Result:=PSmallInt(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;          4: Result:=PLongint(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;          8: Result:=PInt64(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;        end;      end else begin        case DataSize of          1: Result:=PByte(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;          2: Result:=PWord(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;          4: Result:=PLongint(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;          8: Result:=PInt64(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;        end;      end;    ptStatic,    ptVirtual:      begin        if (PropInfo^.PropProcs and 3)=ptStatic then          AMethod.Code:=PropInfo^.GetProc        else          AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;        AMethod.Data:=Instance;        if ((PropInfo^.PropProcs shr 6) and 1)<>0 then begin          case DataSize of            1: Result:=TGetByteProcIndex(AMethod)(PropInfo^.Index);            2: Result:=TGetWordProcIndex(AMethod)(PropInfo^.Index);            4: Result:=TGetIntegerProcIndex(AMethod)(PropInfo^.Index);            8: result:=TGetInt64ProcIndex(AMethod)(PropInfo^.Index)          end;        end else begin          case DataSize of            1: Result:=TGetByteProc(AMethod)();            2: Result:=TGetWordProc(AMethod)();            4: Result:=TGetIntegerProc(AMethod)();            8: result:=TGetInt64Proc(AMethod)();          end;        end;        if Signed then begin          case DataSize of            1: Result:=ShortInt(Result);            2: Result:=SmallInt(Result);          end;        end;      end;  else    raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);  end;end;Procedure SetOrdProp(Instance : TObject;PropInfo : PPropInfo;Value : Int64);type  TSetInt64ProcIndex=procedure(index:longint;i:Int64) of object;  TSetInt64Proc=procedure(i:Int64) of object;  TSetIntegerProcIndex=procedure(index,i:longint) of object;  TSetIntegerProc=procedure(i:longint) of object;var  DataSize: Integer;  AMethod : TMethod;begin  if PropInfo^.PropType^.Kind in [tkInt64,tkQword  { why do we have to handle classes here, see also below? (FK) }{$ifdef cpu64}    ,tkInterface    ,tkInterfaceRaw    ,tkDynArray    ,tkClass{$endif cpu64}    ] then    DataSize := 8  else    DataSize := 4;  if not(PropInfo^.PropType^.Kind in [tkInt64,tkQword,tkClass,tkInterface,tkInterfaceRaw,tkDynArray]) then    begin      { cut off unnecessary stuff }      case GetTypeData(PropInfo^.PropType)^.OrdType of        otSWord,otUWord:          begin            Value:=Value and $ffff;            DataSize := 2;          end;        otSByte,otUByte:          begin            Value:=Value and $ff;            DataSize := 1;          end;       end;    end;  case (PropInfo^.PropProcs shr 2) and 3 of    ptField:      case DataSize of        1: PByte(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Byte(Value);        2: PWord(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Word(Value);        4: PLongint(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Longint(Value);        8: PInt64(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;      end;    ptStatic,    ptVirtual:      begin        if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then          AMethod.Code:=PropInfo^.SetProc        else          AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;        AMethod.Data:=Instance;        if datasize=8 then          begin            if ((PropInfo^.PropProcs shr 6) and 1)<>0 then              TSetInt64ProcIndex(AMethod)(PropInfo^.Index,Value)            else              TSetInt64Proc(AMethod)(Value);          end        else          begin            if ((PropInfo^.PropProcs shr 6) and 1)<>0 then              TSetIntegerProcIndex(AMethod)(PropInfo^.Index,Value)            else              TSetIntegerProc(AMethod)(Value);          end;      end;  else    raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);  end;end;Function GetOrdProp(Instance: TObject; const PropName: string): Int64;begin  Result:=GetOrdProp(Instance,FindPropInfo(Instance,PropName));end;Procedure SetOrdProp(Instance: TObject; const PropName: string;  Value: Int64);begin  SetOrdProp(Instance,FindPropInfo(Instance,PropName),Value);end;Function GetEnumProp(Instance: TObject; Const PropInfo: PPropInfo): string;begin  Result:=GetEnumName(PropInfo^.PropType, GetOrdProp(Instance, PropInfo));end;Function GetEnumProp(Instance: TObject; const PropName: string): string;begin  Result:=GetEnumProp(Instance,FindPropInfo(Instance,PropName));end;Procedure SetEnumProp(Instance: TObject; const PropName: string;  const Value: string);begin  SetEnumProp(Instance,FindPropInfo(Instance,PropName),Value);end;Procedure SetEnumProp(Instance: TObject; Const PropInfo : PPropInfo; const Value: string);Var  PV : Longint;begin  If PropInfo<>Nil then    begin      PV:=GetEnumValue(PropInfo^.PropType, Value);      if (PV<0) then        raise EPropertyError.CreateFmt(SErrUnknownEnumValue, [Value]);      SetOrdProp(Instance, PropInfo,PV);    end;end;{ ---------------------------------------------------------------------  Int64 wrappers  ---------------------------------------------------------------------}Function GetInt64Prop(Instance: TObject; PropInfo: PPropInfo): Int64;begin  Result:=GetOrdProp(Instance,PropInfo);end;procedure SetInt64Prop(Instance: TObject; PropInfo: PPropInfo; const Value: Int64);begin  SetOrdProp(Instance,PropInfo,Value);end;Function GetInt64Prop(Instance: TObject; const PropName: string): Int64;begin  Result:=GetInt64Prop(Instance,FindPropInfo(Instance,PropName));end;Procedure SetInt64Prop(Instance: TObject; const PropName: string; const Value: Int64);begin  SetInt64Prop(Instance,FindPropInfo(Instance,PropName),Value);end;{ ---------------------------------------------------------------------  Set properties  ---------------------------------------------------------------------}Function GetSetProp(Instance: TObject; const PropName: string): string;begin  Result:=GetSetProp(Instance,PropName,False);end;Function GetSetProp(Instance: TObject; const PropName: string; Brackets: Boolean): string;begin  Result:=GetSetProp(Instance,FindPropInfo(Instance,PropName),Brackets);end;Function GetSetProp(Instance: TObject; const PropInfo: PPropInfo; Brackets: Boolean): string;begin  Result:=SetToString(PropInfo,GetOrdProp(Instance,PropInfo),Brackets);end;Procedure SetSetProp(Instance: TObject; const PropName: string; const Value: string);begin  SetSetProp(Instance,FindPropInfo(Instance,PropName),Value);end;Procedure SetSetProp(Instance: TObject; const PropInfo: PPropInfo; const Value: string);begin  SetOrdProp(Instance,PropInfo,StringToSet(PropInfo,Value));end;{ ---------------------------------------------------------------------  Pointer properties - internal only  ---------------------------------------------------------------------}Function GetPointerProp(Instance: TObject; PropInfo : PPropInfo): Pointer;Type  TGetPointerProcIndex = function (index:longint): Pointer of object;  TGetPointerProc = function (): Pointer of object;var  AMethod : TMethod;begin  case (PropInfo^.PropProcs) and 3 of  ptField:    Result := PPointer(Pointer(Instance) + LongWord(PropInfo^.GetProc))^;  ptStatic,  ptVirtual:    begin      if (PropInfo^.PropProcs and 3)=ptStatic then        AMethod.Code:=PropInfo^.GetProc      else        AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;      AMethod.Data:=Instance;      if ((PropInfo^.PropProcs shr 6) and 1)<>0 then        Result:=TGetPointerProcIndex(AMethod)(PropInfo^.Index)      else        Result:=TGetPointerProc(AMethod)();    end;  else    raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);  end;end;Procedure SetPointerProp(Instance: TObject; PropInfo : PPropInfo;  Value: Pointer);type  TSetPointerProcIndex = procedure(index: longint; p: pointer) of object;  TSetPointerProc = procedure(p: pointer) of object;var  AMethod : TMethod;begin  case (PropInfo^.PropProcs shr 2) and 3 of    ptField:      PPointer(Pointer(Instance) + LongWord(PropInfo^.SetProc))^:=Value;    ptStatic,    ptVirtual:      begin        if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then          AMethod.Code:=PropInfo^.SetProc        else          AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;        AMethod.Data:=Instance;        if ((PropInfo^.PropProcs shr 6) and 1)<>0 then          TSetPointerProcIndex(AMethod)(PropInfo^.Index,Value)        else          TSetPointerProc(AMethod)(Value);      end;  else    raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);  end;end;{ ---------------------------------------------------------------------  Object properties  ---------------------------------------------------------------------}Function GetObjectProp(Instance: TObject; const PropName: string): TObject;begin  Result:=GetObjectProp(Instance,PropName,Nil);end;Function GetObjectProp(Instance: TObject; const PropName: string; MinClass: TClass): TObject;begin  Result:=GetObjectProp(Instance,FindPropInfo(Instance,PropName),MinClass);end;Function GetObjectProp(Instance: TObject; PropInfo : PPropInfo): TObject;begin  Result:=GetObjectProp(Instance,PropInfo,Nil);end;Function GetObjectProp(Instance: TObject; PropInfo : PPropInfo; MinClass: TClass): TObject;begin  Result:=TObject(GetPointerProp(Instance,PropInfo));  If (MinClass<>Nil) and (Result<>Nil) Then    If Not Result.InheritsFrom(MinClass) then      Result:=Nil;end;Procedure SetObjectProp(Instance: TObject; const PropName: string;  Value: TObject);begin  SetObjectProp(Instance,FindPropInfo(Instance,PropName),Value);end;Procedure SetObjectProp(Instance: TObject; PropInfo : PPropInfo;  Value: TObject);begin  SetPointerProp(Instance,PropInfo,Pointer(Value));end;Function GetObjectPropClass(Instance: TObject; const PropName: string): TClass;begin  Result:=GetTypeData(FindPropInfo(Instance,PropName,[tkClass])^.PropType)^.ClassType;end;Function  GetObjectPropClass(AClass: TClass; const PropName: string): TClass;begin  Result:=GetTypeData(FindPropInfo(AClass,PropName,[tkClass])^.PropType)^.ClassType;end;{ ---------------------------------------------------------------------    Interface wrapprers  ---------------------------------------------------------------------}function GetInterfaceProp(Instance: TObject; const PropName: string): IInterface;begin  Result:=GetInterfaceProp(Instance,FindPropInfo(Instance,PropName));end;function GetInterfaceProp(Instance: TObject; PropInfo: PPropInfo): IInterface;type  TGetInterfaceProc=function:IInterface of object;  TGetInterfaceProcIndex=function(index:longint):IInterface of object;var  AMethod : TMethod;begin  Result:=nil;  case (PropInfo^.PropProcs) and 3 of    ptField:      Result:=IInterface(PPointer(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^);    ptStatic,    ptVirtual:      begin        if (PropInfo^.PropProcs and 3)=ptStatic then          AMethod.Code:=PropInfo^.GetProc        else          AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;        AMethod.Data:=Instance;        if ((PropInfo^.PropProcs shr 6) and 1)<>0 then          Result:=TGetInterfaceProcIndex(AMethod)(PropInfo^.Index)        else          Result:=TGetInterfaceProc(AMethod)();      end;  else    raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);  end;end;procedure SetInterfaceProp(Instance: TObject; const PropName: string; const Value: IInterface);begin  SetInterfaceProp(Instance,FindPropInfo(Instance,PropName),Value);end;procedure SetInterfaceProp(Instance: TObject; PropInfo: PPropInfo; const Value: IInterface);type  TSetIntfStrProcIndex=procedure(index:longint;const i:IInterface) of object;  TSetIntfStrProc=procedure(i:IInterface) of object;var  AMethod : TMethod;begin  case Propinfo^.PropType^.Kind of    tkInterface:      begin        case (PropInfo^.PropProcs shr 2) and 3 of          ptField:            PInterface(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;          ptStatic,          ptVirtual:            begin              if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then                AMethod.Code:=PropInfo^.SetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                TSetIntfStrProcIndex(AMethod)(PropInfo^.Index,Value)              else                TSetIntfStrProc(AMethod)(Value);            end;        else          raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);        end;      end;    tkInterfaceRaw:      Raise Exception.Create('Cannot set RAW interface from IUnknown interface');  end;end;{ ---------------------------------------------------------------------    RAW (Corba) Interface wrapprers  ---------------------------------------------------------------------}function GetRawInterfaceProp(Instance: TObject; const PropName: string): Pointer;begin  Result:=GetRawInterfaceProp(Instance,FindPropInfo(Instance,PropName));end;function GetRawInterfaceProp(Instance: TObject; PropInfo: PPropInfo): Pointer;begin  Result:=GetPointerProp(Instance,PropInfo);end;procedure SetRawInterfaceProp(Instance: TObject; const PropName: string; const Value: Pointer);begin  SetRawInterfaceProp(Instance,FindPropInfo(Instance,PropName),Value);end;procedure SetRawInterfaceProp(Instance: TObject; PropInfo: PPropInfo; const Value: Pointer);begin  SetPointerProp(Instance,PropInfo,Value);end;{ ---------------------------------------------------------------------  Dynamic array properties  ---------------------------------------------------------------------}function GetDynArrayProp(Instance: TObject; const PropName: string): Pointer;begin  Result:=GetDynArrayProp(Instance,FindPropInfo(Instance,PropName));end;function GetDynArrayProp(Instance: TObject; PropInfo: PPropInfo): Pointer;type  { we need a dynamic array as that type is usually passed differently from    a plain pointer }  TDynArray=array of Byte;  TGetDynArrayProc=function:TDynArray of object;  TGetDynArrayProcIndex=function(index:longint):TDynArray of object;var  AMethod : TMethod;begin  Result:=nil;  if PropInfo^.PropType^.Kind<>tkDynArray then    Exit;  case (PropInfo^.PropProcs) and 3 of    ptField:      Result:=PPointer(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;    ptStatic,    ptVirtual:      begin        if (PropInfo^.PropProcs and 3)=ptStatic then          AMethod.Code:=PropInfo^.GetProc        else          AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;        AMethod.Data:=Instance;        if ((PropInfo^.PropProcs shr 6) and 1)<>0 then          Result:=Pointer(TGetDynArrayProcIndex(AMethod)(PropInfo^.Index))        else          Result:=Pointer(TGetDynArrayProc(AMethod)());      end;  else    raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);  end;end;procedure SetDynArrayProp(Instance: TObject; const PropName: string; const Value: Pointer);begin  SetDynArrayProp(Instance,FindPropInfo(Instance,PropName),Value);end;procedure SetDynArrayProp(Instance: TObject; PropInfo: PPropInfo; const Value: Pointer);type  { we need a dynamic array as that type is usually passed differently from    a plain pointer }  TDynArray=array of Byte;  TSetDynArrayProcIndex=procedure(index:longint;const i:TDynArray) of object;  TSetDynArrayProc=procedure(i:TDynArray) of object;var  AMethod: TMethod;begin  if PropInfo^.PropType^.Kind<>tkDynArray then    Exit;  case (PropInfo^.PropProcs shr 2) and 3 of    ptField:      CopyArray(PPointer(Pointer(Instance)+PtrUInt(PropInfo^.SetProc)), @Value, PropInfo^.PropType, 1);    ptStatic,    ptVirtual:      begin        if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then          AMethod.Code:=PropInfo^.SetProc        else          AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;        AMethod.Data:=Instance;        if ((PropInfo^.PropProcs shr 6) and 1)<>0 then          TSetDynArrayProcIndex(AMethod)(PropInfo^.Index,TDynArray(Value))        else          TSetDynArrayProc(AMethod)(TDynArray(Value));      end;  else    raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);  end;end;{ ---------------------------------------------------------------------  String properties  ---------------------------------------------------------------------}Function GetStrProp(Instance: TObject; PropInfo: PPropInfo): AnsiString;type  TGetShortStrProcIndex=function(index:longint):ShortString of object;  TGetShortStrProc=function():ShortString of object;  TGetAnsiStrProcIndex=function(index:longint):AnsiString of object;  TGetAnsiStrProc=function():AnsiString of object;var  AMethod : TMethod;begin  Result:='';  case Propinfo^.PropType^.Kind of    tkWString:      Result:=AnsiString(GetWideStrProp(Instance,PropInfo));    tkUString:      Result := AnsiString(GetUnicodeStrProp(Instance,PropInfo));    tkSString:      begin        case (PropInfo^.PropProcs) and 3 of          ptField:            Result := PShortString(Pointer(Instance) + LongWord(PropInfo^.GetProc))^;          ptStatic,          ptVirtual:            begin              if (PropInfo^.PropProcs and 3)=ptStatic then                AMethod.Code:=PropInfo^.GetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                Result:=TGetShortStrProcIndex(AMethod)(PropInfo^.Index)              else                Result:=TGetShortStrProc(AMethod)();            end;        else          raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);        end;      end;    tkAString:      begin        case (PropInfo^.PropProcs) and 3 of          ptField:            Result := PAnsiString(Pointer(Instance) + LongWord(PropInfo^.GetProc))^;          ptStatic,          ptVirtual:            begin              if (PropInfo^.PropProcs and 3)=ptStatic then                AMethod.Code:=PropInfo^.GetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                Result:=TGetAnsiStrProcIndex(AMethod)(PropInfo^.Index)              else                Result:=TGetAnsiStrProc(AMethod)();            end;        else          raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);        end;      end;  end;end;Procedure SetStrProp(Instance : TObject;PropInfo : PPropInfo; const Value : AnsiString);type  TSetShortStrProcIndex=procedure(index:longint;const s:ShortString) of object;  TSetShortStrProc=procedure(const s:ShortString) of object;  TSetAnsiStrProcIndex=procedure(index:longint;s:AnsiString) of object;  TSetAnsiStrProc=procedure(s:AnsiString) of object;var  AMethod : TMethod;begin  case Propinfo^.PropType^.Kind of    tkWString:      SetWideStrProp(Instance,PropInfo,WideString(Value));    tkUString:       SetUnicodeStrProp(Instance,PropInfo,UnicodeString(Value));    tkSString:      begin        case (PropInfo^.PropProcs shr 2) and 3 of          ptField:            PShortString(Pointer(Instance) + LongWord(PropInfo^.SetProc))^:=Value;          ptStatic,          ptVirtual:            begin              if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then                AMethod.Code:=PropInfo^.SetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                TSetShortStrProcIndex(AMethod)(PropInfo^.Index,Value)              else                TSetShortStrProc(AMethod)(Value);            end;        else          raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);        end;      end;    tkAString:      begin        case (PropInfo^.PropProcs shr 2) and 3 of          ptField:            PAnsiString(Pointer(Instance) + LongWord(PropInfo^.SetProc))^:=Value;          ptStatic,          ptVirtual:            begin              if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then                AMethod.Code:=PropInfo^.SetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                TSetAnsiStrProcIndex(AMethod)(PropInfo^.Index,Value)              else                TSetAnsiStrProc(AMethod)(Value);            end;          else            raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);        end;      end;  end;end;Function GetStrProp(Instance: TObject; const PropName: string): string;begin  Result:=GetStrProp(Instance,FindPropInfo(Instance,PropName));end;Procedure SetStrProp(Instance: TObject; const PropName: string; const Value: AnsiString);begin  SetStrProp(Instance,FindPropInfo(Instance,PropName),Value);end;Function GetWideStrProp(Instance: TObject; const PropName: string): WideString;begin  Result:=GetWideStrProp(Instance, FindPropInfo(Instance, PropName));end;procedure SetWideStrProp(Instance: TObject; const PropName: string; const Value: WideString);begin  SetWideStrProp(Instance,FindPropInfo(Instance,PropName),Value);end;Function GetWideStrProp(Instance: TObject; PropInfo: PPropInfo): WideString;type  TGetWideStrProcIndex=function(index:longint):WideString of object;  TGetWideStrProc=function():WideString of object;var  AMethod : TMethod;begin  Result:='';  case Propinfo^.PropType^.Kind of    tkSString,tkAString:      Result:=WideString(GetStrProp(Instance,PropInfo));    tkUString :      Result := GetUnicodeStrProp(Instance,PropInfo);    tkWString:      begin        case (PropInfo^.PropProcs) and 3 of          ptField:            Result := PWideString(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;          ptStatic,          ptVirtual:            begin              if (PropInfo^.PropProcs and 3)=ptStatic then                AMethod.Code:=PropInfo^.GetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                Result:=TGetWideStrProcIndex(AMethod)(PropInfo^.Index)              else                Result:=TGetWideStrProc(AMethod)();            end;        else          raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);        end;      end;  end;end;Procedure SetWideStrProp(Instance: TObject; PropInfo: PPropInfo; const Value: WideString);type  TSetWideStrProcIndex=procedure(index:longint;s:WideString) of object;  TSetWideStrProc=procedure(s:WideString) of object;var  AMethod : TMethod;begin  case Propinfo^.PropType^.Kind of    tkSString,tkAString:       SetStrProp(Instance,PropInfo,AnsiString(Value));    tkUString:       SetUnicodeStrProp(Instance,PropInfo,Value);    tkWString:      begin        case (PropInfo^.PropProcs shr 2) and 3 of          ptField:            PWideString(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;          ptStatic,          ptVirtual:            begin              if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then                AMethod.Code:=PropInfo^.SetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                TSetWideStrProcIndex(AMethod)(PropInfo^.Index,Value)              else                TSetWideStrProc(AMethod)(Value);            end;        else          raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);        end;      end;  end;end;Function GetUnicodeStrProp(Instance: TObject; const PropName: string): UnicodeString;begin  Result:=GetUnicodeStrProp(Instance, FindPropInfo(Instance, PropName));end;procedure SetUnicodeStrProp(Instance: TObject; const PropName: string; const Value: UnicodeString);begin  SetUnicodeStrProp(Instance,FindPropInfo(Instance,PropName),Value);end;Function GetUnicodeStrProp(Instance: TObject; PropInfo: PPropInfo): UnicodeString;type  TGetUnicodeStrProcIndex=function(index:longint):UnicodeString of object;  TGetUnicodeStrProc=function():UnicodeString of object;var  AMethod : TMethod;begin  Result:='';  case Propinfo^.PropType^.Kind of    tkSString,tkAString:      Result:=UnicodeString(GetStrProp(Instance,PropInfo));    tkWString:      Result:=GetWideStrProp(Instance,PropInfo);    tkUString:      begin        case (PropInfo^.PropProcs) and 3 of          ptField:            Result := PUnicodeString(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;          ptStatic,          ptVirtual:            begin              if (PropInfo^.PropProcs and 3)=ptStatic then                AMethod.Code:=PropInfo^.GetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                Result:=TGetUnicodeStrProcIndex(AMethod)(PropInfo^.Index)              else                Result:=TGetUnicodeStrProc(AMethod)();            end;        else          raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);        end;      end;  end;end;Procedure SetUnicodeStrProp(Instance: TObject; PropInfo: PPropInfo; const Value: UnicodeString);type  TSetUnicodeStrProcIndex=procedure(index:longint;s:UnicodeString) of object;  TSetUnicodeStrProc=procedure(s:UnicodeString) of object;var  AMethod : TMethod;begin  case Propinfo^.PropType^.Kind of    tkSString,tkAString:       SetStrProp(Instance,PropInfo,AnsiString(Value));    tkWString:       SetWideStrProp(Instance,PropInfo,Value);    tkUString:      begin        case (PropInfo^.PropProcs shr 2) and 3 of          ptField:            PUnicodeString(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;          ptStatic,          ptVirtual:            begin              if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then                AMethod.Code:=PropInfo^.SetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                TSetUnicodeStrProcIndex(AMethod)(PropInfo^.Index,Value)              else                TSetUnicodeStrProc(AMethod)(Value);            end;        else          raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);        end;      end;  end;end;function GetRawbyteStrProp(Instance: TObject; PropInfo: PPropInfo): RawByteString;type  TGetRawByteStrProcIndex=function(index:longint): RawByteString of object;  TGetRawByteStrProc=function():RawByteString of object;var  AMethod : TMethod;begin  Result:='';  case Propinfo^.PropType^.Kind of    tkWString:      Result:=RawByteString(GetWideStrProp(Instance,PropInfo));    tkUString:      Result:=RawByteString(GetUnicodeStrProp(Instance,PropInfo));    tkSString:      Result:=RawByteString(GetStrProp(Instance,PropInfo));    tkAString:      begin        case (PropInfo^.PropProcs) and 3 of          ptField:            Result := PAnsiString(Pointer(Instance) + LongWord(PropInfo^.GetProc))^;          ptStatic,          ptVirtual:            begin              if (PropInfo^.PropProcs and 3)=ptStatic then                AMethod.Code:=PropInfo^.GetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                Result:=TGetRawByteStrProcIndex(AMethod)(PropInfo^.Index)              else                Result:=TGetRawByteStrProc(AMethod)();            end;          else            raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);        end;      end;  end;end;function GetRawByteStrProp(Instance: TObject; const PropName: string): RawByteString;begin  Result:=GetRawByteStrProp(Instance,FindPropInfo(Instance,PropName));end;procedure SetRawByteStrProp(Instance: TObject; PropInfo: PPropInfo; const Value: RawByteString);type  TSetRawByteStrProcIndex=procedure(index:longint;s:RawByteString) of object;  TSetRawByteStrProc=procedure(s:RawByteString) of object;var  AMethod : TMethod;begin  case Propinfo^.PropType^.Kind of    tkWString:      SetWideStrProp(Instance,PropInfo,WideString(Value));    tkUString:       SetUnicodeStrProp(Instance,PropInfo,UnicodeString(Value));    tkSString:      SetStrProp(Instance,PropInfo,Value); // Not 100% sure about this.    tkAString:      begin        case (PropInfo^.PropProcs shr 2) and 3 of          ptField:            PAnsiString(Pointer(Instance) + LongWord(PropInfo^.SetProc))^:=Value;          ptStatic,          ptVirtual:            begin              if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then                AMethod.Code:=PropInfo^.SetProc              else                AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;              AMethod.Data:=Instance;              if ((PropInfo^.PropProcs shr 6) and 1)<>0 then                TSetRawByteStrProcIndex(AMethod)(PropInfo^.Index,Value)              else                TSetRawByteStrProc(AMethod)(Value);            end;        else          raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);        end;      end;  end;end;procedure SetRawByteStrProp(Instance: TObject; const PropName: string; const Value: RawByteString);begin  SetRawByteStrProp(Instance,FindPropInfo(Instance,PropName),Value);end;{$ifndef FPUNONE}{ ---------------------------------------------------------------------  Float properties  ---------------------------------------------------------------------}function GetFloatProp(Instance : TObject;PropInfo : PPropInfo) : Extended;type  TGetExtendedProc = function:Extended of object;  TGetExtendedProcIndex = function(Index: integer): Extended of object;  TGetDoubleProc = function:Double of object;  TGetDoubleProcIndex = function(Index: integer): Double of object;  TGetSingleProc = function:Single of object;  TGetSingleProcIndex = function(Index: integer):Single of object;  TGetCurrencyProc = function : Currency of object;  TGetCurrencyProcIndex = function(Index: integer) : Currency of object;var  AMethod : TMethod;begin  Result:=0.0;  case PropInfo^.PropProcs and 3 of    ptField:      Case GetTypeData(PropInfo^.PropType)^.FloatType of       ftSingle:         Result:=PSingle(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;       ftDouble:         Result:=PDouble(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;       ftExtended:         Result:=PExtended(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;       ftcomp:         Result:=PComp(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;       ftcurr:         Result:=PCurrency(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;       end;    ptStatic,    ptVirtual:      begin        if (PropInfo^.PropProcs and 3)=ptStatic then          AMethod.Code:=PropInfo^.GetProc        else          AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;        AMethod.Data:=Instance;        Case GetTypeData(PropInfo^.PropType)^.FloatType of          ftSingle:            if ((PropInfo^.PropProcs shr 6) and 1)=0 then              Result:=TGetSingleProc(AMethod)()            else              Result:=TGetSingleProcIndex(AMethod)(PropInfo^.Index);          ftDouble:            if ((PropInfo^.PropProcs shr 6) and 1)=0 then              Result:=TGetDoubleProc(AMethod)()            else              Result:=TGetDoubleProcIndex(AMethod)(PropInfo^.Index);          ftExtended:            if ((PropInfo^.PropProcs shr 6) and 1)=0 then              Result:=TGetExtendedProc(AMethod)()            else              Result:=TGetExtendedProcIndex(AMethod)(PropInfo^.Index);          ftCurr:            if ((PropInfo^.PropProcs shr 6) and 1)=0 then              Result:=TGetCurrencyProc(AMethod)()            else              Result:=TGetCurrencyProcIndex(AMethod)(PropInfo^.Index);        end;      end;  else    raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);  end;end;Procedure SetFloatProp(Instance : TObject;PropInfo : PPropInfo; Value : Extended);type  TSetExtendedProc = procedure(const AValue: Extended) of object;  TSetExtendedProcIndex = procedure(Index: integer; AValue: Extended) of object;  TSetDoubleProc = procedure(const AValue: Double) of object;  TSetDoubleProcIndex = procedure(Index: integer; AValue: Double) of object;  TSetSingleProc = procedure(const AValue: Single) of object;  TSetSingleProcIndex = procedure(Index: integer; AValue: Single) of object;  TSetCurrencyProc = procedure(const AValue: Currency) of object;  TSetCurrencyProcIndex = procedure(Index: integer;  AValue: Currency) of object;Var  AMethod : TMethod;begin  case (PropInfo^.PropProcs shr 2) and 3 of    ptfield:      Case GetTypeData(PropInfo^.PropType)^.FloatType of        ftSingle:          PSingle(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;        ftDouble:          PDouble(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;        ftExtended:          PExtended(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;{$ifdef FPC_COMP_IS_INT64}        ftComp:          PComp(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=trunc(Value);{$else FPC_COMP_IS_INT64}        ftComp:          PComp(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Comp(Value);{$endif FPC_COMP_IS_INT64}        ftCurr:          PCurrency(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;       end;    ptStatic,    ptVirtual:      begin        if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then          AMethod.Code:=PropInfo^.SetProc        else          AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;        AMethod.Data:=Instance;        Case GetTypeData(PropInfo^.PropType)^.FloatType of          ftSingle:            if ((PropInfo^.PropProcs shr 6) and 1)=0 then              TSetSingleProc(AMethod)(Value)            else              TSetSingleProcIndex(AMethod)(PropInfo^.Index,Value);          ftDouble:            if ((PropInfo^.PropProcs shr 6) and 1)=0 then              TSetDoubleProc(AMethod)(Value)            else              TSetDoubleProcIndex(AMethod)(PropInfo^.Index,Value);          ftExtended:            if ((PropInfo^.PropProcs shr 6) and 1)=0 then              TSetExtendedProc(AMethod)(Value)            else              TSetExtendedProcIndex(AMethod)(PropInfo^.Index,Value);          ftCurr:            if ((PropInfo^.PropProcs shr 6) and 1)=0 then              TSetCurrencyProc(AMethod)(Value)            else              TSetCurrencyProcIndex(AMethod)(PropInfo^.Index,Value);        end;      end;  else    raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);  end;end;function GetFloatProp(Instance: TObject; const PropName: string): Extended;begin  Result:=GetFloatProp(Instance,FindPropInfo(Instance,PropName))end;Procedure SetFloatProp(Instance: TObject; const PropName: string;  Value: Extended);begin  SetFloatProp(Instance,FindPropInfo(Instance,PropName),Value);end;{$endif}{ ---------------------------------------------------------------------  Method properties  ---------------------------------------------------------------------}Function GetMethodProp(Instance : TObject;PropInfo : PPropInfo) : TMethod;type  TGetMethodProcIndex=function(Index: Longint): TMethod of object;  TGetMethodProc=function(): TMethod of object;var  value: PMethod;  AMethod : TMethod;begin  Result.Code:=nil;  Result.Data:=nil;  case (PropInfo^.PropProcs) and 3 of    ptField:      begin        Value:=PMethod(Pointer(Instance)+PtrUInt(PropInfo^.GetProc));        if Value<>nil then          Result:=Value^;      end;    ptStatic,    ptVirtual:      begin        if (PropInfo^.PropProcs and 3)=ptStatic then          AMethod.Code:=PropInfo^.GetProc        else          AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;        AMethod.Data:=Instance;        if ((PropInfo^.PropProcs shr 6) and 1)<>0 then          Result:=TGetMethodProcIndex(AMethod)(PropInfo^.Index)        else          Result:=TGetMethodProc(AMethod)();      end;  else    raise EPropertyError.CreateFmt(SErrCannotReadProperty, [PropInfo^.Name]);  end;end;Procedure SetMethodProp(Instance : TObject;PropInfo : PPropInfo; const Value : TMethod);type  TSetMethodProcIndex=procedure(index:longint;p:TMethod) of object;  TSetMethodProc=procedure(p:TMethod) of object;var  AMethod : TMethod;begin  case (PropInfo^.PropProcs shr 2) and 3 of    ptField:      PMethod(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^ := Value;    ptStatic,    ptVirtual:      begin        if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then          AMethod.Code:=PropInfo^.SetProc        else          AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;        AMethod.Data:=Instance;        if ((PropInfo^.PropProcs shr 6) and 1)<>0 then          TSetMethodProcIndex(AMethod)(PropInfo^.Index,Value)        else          TSetMethodProc(AMethod)(Value);      end;  else    raise EPropertyError.CreateFmt(SErrCannotWriteToProperty, [PropInfo^.Name]);  end;end;Function GetMethodProp(Instance: TObject; const PropName: string): TMethod;begin  Result:=GetMethodProp(Instance,FindPropInfo(Instance,PropName));end;Procedure SetMethodProp(Instance: TObject; const PropName: string;  const Value: TMethod);begin  SetMethodProp(Instance,FindPropInfo(Instance,PropName),Value);end;{ ---------------------------------------------------------------------  Variant properties  ---------------------------------------------------------------------}Procedure CheckVariantEvent(P : CodePointer);begin  If (P=Nil) then    Raise Exception.Create(SErrNoVariantSupport);end;Function GetVariantProp(Instance : TObject;PropInfo : PPropInfo): Variant;begin  CheckVariantEvent(CodePointer(OnGetVariantProp));  Result:=OnGetVariantProp(Instance,PropInfo);end;Procedure SetVariantProp(Instance : TObject;PropInfo : PPropInfo; const Value: Variant);begin  CheckVariantEvent(CodePointer(OnSetVariantProp));  OnSetVariantProp(Instance,PropInfo,Value);end;Function GetVariantProp(Instance: TObject; const PropName: string): Variant;begin  Result:=GetVariantProp(Instance,FindPropInfo(Instance,PropName));end;Procedure SetVariantProp(Instance: TObject; const PropName: string;  const Value: Variant);begin  SetVariantprop(instance,FindpropInfo(Instance,PropName),Value);end;{ ---------------------------------------------------------------------  All properties through variant.  ---------------------------------------------------------------------}Function GetPropValue(Instance: TObject; const PropName: string): Variant;begin  Result := GetPropValue(Instance,FindPropInfo(Instance, PropName));end;Function GetPropValue(Instance: TObject; const PropName: string; PreferStrings: Boolean): Variant;begin  Result := GetPropValue(Instance,FindPropInfo(Instance, PropName),PreferStrings);end;Function GetPropValue(Instance: TObject; PropInfo: PPropInfo): Variant;begin  Result := GetPropValue(Instance, PropInfo, True);end;Function GetPropValue(Instance: TObject; PropInfo: PPropInfo; PreferStrings: Boolean): Variant;begin  CheckVariantEvent(CodePointer(OnGetPropValue));  Result:=OnGetPropValue(Instance,PropInfo,PreferStrings);end;Procedure SetPropValue(Instance: TObject; const PropName: string;  const Value: Variant);begin  SetPropValue(Instance, FindPropInfo(Instance, PropName), Value);end;Procedure SetPropValue(Instance: TObject; PropInfo: PPropInfo;  const Value: Variant);begin  CheckVariantEvent(CodePointer(OnSetPropValue));  OnSetPropValue(Instance,PropInfo,Value);end;{ ---------------------------------------------------------------------  Easy access methods that appeared in Delphi 5  ---------------------------------------------------------------------}Function IsPublishedProp(Instance: TObject; const PropName: string): Boolean;begin  Result:=GetPropInfo(Instance,PropName)<>Nil;end;Function IsPublishedProp(AClass: TClass; const PropName: string): Boolean;begin  Result:=GetPropInfo(AClass,PropName)<>Nil;end;Function PropIsType(Instance: TObject; const PropName: string; TypeKind: TTypeKind): Boolean;begin  Result:=PropType(Instance,PropName)=TypeKindend;Function PropIsType(AClass: TClass; const PropName: string; TypeKind: TTypeKind): Boolean;begin  Result:=PropType(AClass,PropName)=TypeKindend;Function PropType(Instance: TObject; const PropName: string): TTypeKind;begin  Result:=FindPropInfo(Instance,PropName)^.PropType^.Kind;end;Function PropType(AClass: TClass; const PropName: string): TTypeKind;begin  Result:=FindPropInfo(AClass,PropName)^.PropType^.Kind;end;Function IsStoredProp(Instance: TObject; const PropName: string): Boolean;begin  Result:=IsStoredProp(instance,FindPropInfo(Instance,PropName));end;{ TParameterLocation }function TParameterLocation.GetReference: Boolean;begin  Result := (LocType and $80) <> 0;end;function TParameterLocation.GetRegType: TRegisterType;begin  Result := TRegisterType(LocType and $7F);end;function TParameterLocation.GetShiftVal: Int8;begin  if GetReference then begin    if Offset < Low(Int8) then      Result := Low(Int8)    else if Offset > High(Int8) then      Result := High(Int8)    else      Result := Offset;  end else    Result := 0;end;{ TParameterLocations }function TParameterLocations.GetLocation(aIndex: Byte): PParameterLocation;begin  if aIndex >= Count then    Result := Nil  else    Result := PParameterLocation(PByte(aligntoptr(PByte(@Count) + SizeOf(Count))) + SizeOf(TParameterLocation) * aIndex);end;function TParameterLocations.GetTail: Pointer;begin  Result := PByte(aligntoptr(PByte(@Count) + SizeOf(Count))) + SizeOf(TParameterLocation) * Count;end;{ TProcedureParam }function TProcedureParam.GetParamType: PTypeInfo;begin  Result := DerefTypeInfoPtr(ParamTypeRef);end;function TProcedureParam.GetFlags: Byte;begin  Result := PByte(@ParamFlags)^;end;{ TManagedField }function TManagedField.GetTypeRef: PTypeInfo;begin  Result := DerefTypeInfoPtr(TypeRefRef);end;{ TArrayTypeData }function TArrayTypeData.GetElType: PTypeInfo;begin  Result := DerefTypeInfoPtr(ElTypeRef);end;function TArrayTypeData.GetDims(aIndex: Byte): PTypeInfo;begin  Result := DerefTypeInfoPtr(DimsRef[aIndex]);end;{ TProcedureSignature }function TProcedureSignature.GetResultType: PTypeInfo;begin  Result := DerefTypeInfoPtr(ResultTypeRef);end;function TProcedureSignature.GetParam(ParamIndex: Integer): PProcedureParam;begin  if (ParamIndex<0)or(ParamIndex>=ParamCount) then    Exit(nil);  Result := PProcedureParam(PByte(@Flags) + SizeOf(Self));  while ParamIndex > 0 do    begin      Result := PProcedureParam(aligntoptr((PByte(@Result^.Name) + (Length(Result^.Name) + 1) * SizeOf(AnsiChar))));      dec(ParamIndex);    end;end;{ TVmtMethodParam }function TVmtMethodParam.GetTail: Pointer;begin  Result := PByte(@ParaLocs) + SizeOf(ParaLocs);end;function TVmtMethodParam.GetNext: PVmtMethodParam;begin  Result := PVmtMethodParam(aligntoptr(Tail));end;function TVmtMethodParam.GetName: ShortString;begin  Result := NamePtr^;end;{ TIntfMethodEntry }function TIntfMethodEntry.GetParam(Index: Word): PVmtMethodParam;begin  if Index >= ParamCount then    Result := Nil  else    Result := PVmtMethodParam(PByte(aligntoptr(PByte(@NamePtr) + SizeOf(NamePtr))) + Index * PtrUInt(aligntoptr(Pointer(SizeOf(TVmtMethodParam)))));end;function TIntfMethodEntry.GetResultLocs: PParameterLocations;begin  if not Assigned(ResultType) then    Result := Nil  else    Result := PParameterLocations(PByte(aligntoptr(PByte(@NamePtr) + SizeOf(NamePtr))) + ParamCount * PtrUInt(aligntoptr(Pointer(SizeOf(TVmtMethodParam)))));end;function TIntfMethodEntry.GetTail: Pointer;begin  Result := PByte(@NamePtr) + SizeOf(NamePtr);  if ParamCount > 0 then    Result := PByte(aligntoptr(Result)) + ParamCount * PtrUInt(aligntoptr(Pointer(SizeOf(TVmtMethodParam))));  if Assigned(ResultType) then    Result := PByte(aligntoptr(Result)) + SizeOf(PParameterLocations);end;function TIntfMethodEntry.GetNext: PIntfMethodEntry;begin  Result := PIntfMethodEntry(aligntoptr(Tail));end;function TIntfMethodEntry.GetName: ShortString;begin  Result := NamePtr^;end;{ TIntfMethodTable }function TIntfMethodTable.GetMethod(Index: Word): PIntfMethodEntry;begin  if (RTTICount = $FFFF) or (Index >= RTTICount) then    Result := Nil  else    begin      Result := aligntoptr(PIntfMethodEntry(PByte(@RTTICount) + SizeOf(RTTICount)));      while Index > 0 do        begin          Result := Result^.Next;          Dec(Index);        end;    end;end;{ TVmtMethodTable }function TVmtMethodTable.GetEntry(Index: LongWord): PVmtMethodEntry;begin  Result := PVmtMethodEntry(@Entries[0]) + Index;end;{ TVmtFieldTable }function TVmtFieldTable.GetField(aIndex: Word): PVmtFieldEntry;var  c: Word;begin  if aIndex >= Count then    Exit(Nil);  c := aIndex;  Result := @Fields;  while c > 0 do begin    Result := Result^.Next;    Dec(c);  end;end;{ TVmtFieldEntry }function TVmtFieldEntry.GetNext: PVmtFieldEntry;begin  Result := aligntoptr(Tail);end;function TVmtFieldEntry.GetTail: Pointer;begin  Result := PByte(@Name) + Length(Name) + SizeOf(Byte);end;{ TInterfaceData }function TInterfaceData.GetUnitName: ShortString;begin  Result := UnitNameField;end;function TInterfaceData.GetPropertyTable: PPropData;var  p: PByte;begin  p := PByte(@UnitNameField[0]) + SizeOf(UnitNameField[0]) + Length(UnitNameField);  Result := AlignTypeData(p);end;function TInterfaceData.GetMethodTable: PIntfMethodTable;begin  Result := aligntoptr(PropertyTable^.Tail);end;{ TInterfaceRawData }function TInterfaceRawData.GetUnitName: ShortString;begin  Result := UnitNameField;end;function TInterfaceRawData.GetIIDStr: ShortString;begin  Result := PShortString(AlignTypeData(PByte(@UnitNameField[0]) + SizeOf(UnitNameField[0]) + Length(UnitNameField)))^;end;function TInterfaceRawData.GetPropertyTable: PPropData;var  p: PByte;begin  p := AlignTypeData(PByte(@UnitNameField[0]) + SizeOf(UnitNameField[0]) + Length(UnitNameField));  p := p + SizeOf(p^) + p^;  Result := aligntoptr(p);end;function TInterfaceRawData.GetMethodTable: PIntfMethodTable;begin  Result := aligntoptr(PropertyTable^.Tail);end;{ TClassData }function TClassData.GetUnitName: ShortString;begin  Result := UnitNameField;end;function TClassData.GetPropertyTable: PPropData;var  p: PByte;begin  p := PByte(@UnitNameField[0]) + SizeOf(UnitNameField[0]) + Length(UnitNameField);  Result := AlignToPtr(p);end;{ TTypeData }function TTypeData.GetBaseType: PTypeInfo;begin  Result := DerefTypeInfoPtr(BaseTypeRef);end;function TTypeData.GetCompType: PTypeInfo;begin  Result := DerefTypeInfoPtr(CompTypeRef);end;function TTypeData.GetParentInfo: PTypeInfo;begin  Result := DerefTypeInfoPtr(ParentInfoRef);end;{$ifndef VER3_0}function TTypeData.GetRecInitData: PRecInitData;begin  Result := PRecInitData(aligntoptr(PTypeData(RecInitInfo+2+PByte(RecInitInfo+1)^)));end;{$endif}function TTypeData.GetHelperParent: PTypeInfo;begin  Result := DerefTypeInfoPtr(HelperParentRef);end;function TTypeData.GetExtendedInfo: PTypeInfo;begin  Result := DerefTypeInfoPtr(ExtendedInfoRef);end;function TTypeData.GetIntfParent: PTypeInfo;begin  Result := DerefTypeInfoPtr(IntfParentRef);end;function TTypeData.GetRawIntfParent: PTypeInfo;begin  Result := DerefTypeInfoPtr(RawIntfParentRef);end;function TTypeData.GetIIDStr: ShortString;begin  Result := PShortString(AlignTypeData(Pointer(@RawIntfUnit) + Length(RawIntfUnit) + 1))^;end;function TTypeData.GetElType: PTypeInfo;begin  Result := DerefTypeInfoPtr(elTypeRef);end;function TTypeData.GetElType2: PTypeInfo;begin  Result := DerefTypeInfoPtr(elType2Ref);end;function TTypeData.GetInstanceType: PTypeInfo;begin  Result := DerefTypeInfoPtr(InstanceTypeRef);end;function TTypeData.GetRefType: PTypeInfo;begin  Result := DerefTypeInfoPtr(RefTypeRef);end;{ TPropData }function TPropData.GetProp(Index: Word): PPropInfo;begin  if Index >= PropCount then    Result := Nil  else    begin      Result := PPropInfo(aligntoptr(PByte(@PropCount) + SizeOf(PropCount)));      while Index > 0 do        begin          Result := aligntoptr(Result^.Tail);          Dec(Index);        end;    end;end;function TPropData.GetTail: Pointer;begin  if PropCount = 0 then    Result := PByte(@PropCount) + SizeOf(PropCount)  else    Result := Prop[PropCount - 1]^.Tail;end;{ TPropInfo }function TPropInfo.GetPropType: PTypeInfo;begin  Result := DerefTypeInfoPtr(PropTypeRef);end;function TPropInfo.GetTail: Pointer;begin  Result := PByte(@Name[0]) + SizeOf(Name[0]) + Length(Name);end;function TPropInfo.GetNext: PPropInfo;begin  Result := PPropInfo(aligntoptr(Tail));end;type  TElementAlias = record    Ordinal : Integer;    Alias : string;  end;  TElementAliasArray = Array of TElementAlias;  PElementAliasArray = ^TElementAliasArray;  TEnumeratedAliases = record    TypeInfo: PTypeInfo;    Aliases: TElementAliasArray;  end;  TEnumeratedAliasesArray = Array of TEnumeratedAliases;Var  EnumeratedAliases : TEnumeratedAliasesArray;Function IndexOfEnumeratedAliases(aTypeInfo : PTypeInfo) : integer;begin  Result:=Length(EnumeratedAliases)-1;  while (Result>=0) and (EnumeratedAliases[Result].TypeInfo<>aTypeInfo) do    Dec(Result);end;Function GetEnumeratedAliases(aTypeInfo : PTypeInfo) : PElementAliasArray;Var  I : integer;begin  I:=IndexOfEnumeratedAliases(aTypeInfo);  if I=-1 then    Result:=Nil  else    Result:=@EnumeratedAliases[i].Aliasesend;Function AddEnumeratedAliases(aTypeInfo : PTypeInfo) : PElementAliasArray;Var  L : Integer;begin  L:=Length(EnumeratedAliases);  SetLength(EnumeratedAliases,L+1);  EnumeratedAliases[L].TypeInfo:=aTypeInfo;  Result:=@EnumeratedAliases[L].Aliases;end;procedure RemoveEnumElementAliases(aTypeInfo: PTypeInfo);Var  I,L : integer;  A : TEnumeratedAliases;begin  I:=IndexOfEnumeratedAliases(aTypeInfo);  if I=-1 then    exit;  A:=EnumeratedAliases[i];  A.Aliases:=Nil;  A.TypeInfo:=Nil;  L:=Length(EnumeratedAliases)-1;  EnumeratedAliases[i]:=EnumeratedAliases[L];  EnumeratedAliases[L]:=A;  SetLength(EnumeratedAliases,L);end;Resourcestring  SErrNotAnEnumerated = 'Type information points to non-enumerated type';  SErrInvalidEnumeratedCount = 'Invalid number of enumerated values';  SErrDuplicateEnumerated = 'Duplicate alias for enumerated value';procedure AddEnumElementAliases(aTypeInfo: PTypeInfo; const aNames: array of string; aStartValue: Integer = 0);var  Aliases: PElementAliasArray;  A : TElementAliasArray;  L, I, J : Integer;  N : String;  PT : PTypeData;begin  if (aTypeInfo^.Kind<>tkEnumeration) then    raise EArgumentException.Create(SErrNotAnEnumerated);  PT:=GetTypeData(aTypeInfo);  if (High(aNames)=-1) or ((aStartValue+High(aNames))> PT^.MaxValue) then    raise EArgumentException.Create(SErrInvalidEnumeratedCount);  Aliases:=GetEnumeratedAliases(aTypeInfo);  if (Aliases=Nil) then    Aliases:=AddEnumeratedAliases(aTypeInfo);  A:=Aliases^;  I:=0;  L:=Length(a);  SetLength(a,L+High(aNames)+1);  try    for N in aNames do      begin      for J:=0 to (L+I)-1 do        if SameText(N,A[J].Alias) then          raise EArgumentException.Create(SErrDuplicateEnumerated);      with A[L+I] do        begin        Ordinal:=aStartValue+I;        alias:=N;        end;      Inc(I);      end;  finally    // In case of exception, we need to correct the length.    if Length(A)<>I+L then      SetLength(A,I+L);    Aliases^:=A;  end;end;function GetEnumeratedAliasValue(aTypeInfo: PTypeInfo; const aName: string): Integer;var  I : Integer;  Aliases: PElementAliasArray;begin  Result:=-1;  Aliases:=GetEnumeratedAliases(aTypeInfo);  if (Aliases=Nil) then    Exit;  I:=Length(Aliases^)-1;  While (Result=-1) and (I>=0) do    begin    if SameText(Aliases^[I].Alias, aName) then      Result:=Aliases^[I].Ordinal;    Dec(I);    end;end;end.
 |