variants.pp 127 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551
  1. {
  2. This include file contains the variants
  3. support for FPC
  4. This file is part of the Free Pascal run time library.
  5. Copyright (c) 2001-2005 by the Free Pascal development team
  6. See the file COPYING.FPC, included in this distribution,
  7. for details about the copyright.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  11. **********************************************************************}
  12. {$IFDEF fpc}
  13. {$mode objfpc}
  14. {$ENDIF}
  15. {$h+}
  16. { Using inlining for small system functions/wrappers }
  17. {$inline on}
  18. {$define VARIANTINLINE}
  19. unit variants;
  20. interface
  21. uses
  22. sysutils,sysconst,rtlconsts,typinfo;
  23. type
  24. EVariantParamNotFoundError = class(EVariantError);
  25. EVariantInvalidOpError = class(EVariantError);
  26. EVariantTypeCastError = class(EVariantError);
  27. EVariantOverflowError = class(EVariantError);
  28. EVariantInvalidArgError = class(EVariantError);
  29. EVariantBadVarTypeError = class(EVariantError);
  30. EVariantBadIndexError = class(EVariantError);
  31. EVariantArrayLockedError = class(EVariantError);
  32. EVariantNotAnArrayError = class(EVariantError);
  33. EVariantArrayCreateError = class(EVariantError);
  34. EVariantNotImplError = class(EVariantError);
  35. EVariantOutOfMemoryError = class(EVariantError);
  36. EVariantUnexpectedError = class(EVariantError);
  37. EVariantDispatchError = class(EVariantError);
  38. EVariantRangeCheckError = class(EVariantOverflowError);
  39. EVariantInvalidNullOpError = class(EVariantInvalidOpError);
  40. TVariantRelationship = (vrEqual, vrLessThan, vrGreaterThan, vrNotEqual);
  41. TNullCompareRule = (ncrError, ncrStrict, ncrLoose);
  42. TBooleanToStringRule = (bsrAsIs, bsrLower, bsrUpper);
  43. Const
  44. OrdinalVarTypes = [varSmallInt, varInteger, varBoolean, varShortInt,
  45. varByte, varWord,varLongWord,varInt64];
  46. FloatVarTypes = [
  47. {$ifndef FPUNONE}
  48. varSingle, varDouble,
  49. {$endif}
  50. varCurrency];
  51. { Variant support procedures and functions }
  52. function VarType(const V: Variant): TVarType; inline;
  53. function VarTypeDeRef(const V: Variant): TVarType; overload;
  54. function VarTypeDeRef(const V: TVarData): TVarType; overload; inline;
  55. function VarAsType(const V: Variant; aVarType: TVarType): Variant;
  56. function VarIsType(const V: Variant; aVarType: TVarType): Boolean; overload; inline;
  57. function VarIsType(const V: Variant; const AVarTypes: array of TVarType): Boolean; overload;
  58. function VarIsByRef(const V: Variant): Boolean; inline;
  59. function VarIsEmpty(const V: Variant): Boolean; inline;
  60. procedure VarCheckEmpty(const V: Variant); inline;
  61. function VarIsNull(const V: Variant): Boolean; inline;
  62. function VarIsClear(const V: Variant): Boolean; inline;
  63. function VarIsCustom(const V: Variant): Boolean; inline;
  64. function VarIsOrdinal(const V: Variant): Boolean; inline;
  65. function VarIsFloat(const V: Variant): Boolean; inline;
  66. function VarIsNumeric(const V: Variant): Boolean; inline;
  67. function VarIsStr(const V: Variant): Boolean;
  68. function VarToStr(const V: Variant): string;
  69. function VarToStrDef(const V: Variant; const ADefault: string): string;
  70. function VarToWideStr(const V: Variant): WideString;
  71. function VarToWideStrDef(const V: Variant; const ADefault: WideString): WideString;
  72. function VarToUnicodeStr(const V: Variant): UnicodeString;
  73. function VarToUnicodeStrDef(const V: Variant; const ADefault: UnicodeString): UnicodeString;
  74. {$ifndef FPUNONE}
  75. function VarToDateTime(const V: Variant): TDateTime;
  76. function VarFromDateTime(const DateTime: TDateTime): Variant;
  77. {$endif}
  78. function VarInRange(const AValue, AMin, AMax: Variant): Boolean;
  79. function VarEnsureRange(const AValue, AMin, AMax: Variant): Variant;
  80. function VarSameValue(const A, B: Variant): Boolean;
  81. function VarCompareValue(const A, B: Variant): TVariantRelationship;
  82. function VarIsEmptyParam(const V: Variant): Boolean; inline;
  83. procedure VarClear(var V: Variant);{$IFDEF VARIANTINLINE}inline;{$ENDIF VARIANTINLINE}
  84. procedure VarClear(var V: OleVariant);{$IFDEF VARIANTINLINE}inline;{$ENDIF VARIANTINLINE}
  85. procedure SetClearVarToEmptyParam(var V: TVarData);
  86. function VarIsError(const V: Variant; out AResult: HRESULT): Boolean;
  87. function VarIsError(const V: Variant): Boolean; inline;
  88. function VarAsError(AResult: HRESULT): Variant;
  89. function VarSupports(const V: Variant; const IID: TGUID; out Intf): Boolean;
  90. function VarSupports(const V: Variant; const IID: TGUID): Boolean;
  91. { Variant copy support }
  92. procedure VarCopyNoInd(var Dest: Variant; const Source: Variant);
  93. { Variant array support procedures and functions }
  94. function VarArrayCreate(const Bounds: array of SizeInt; aVarType: TVarType): Variant;
  95. function VarArrayCreate(const Bounds: PVarArrayBoundArray; Dims : SizeInt; aVarType: TVarType): Variant;
  96. function VarArrayOf(const Values: array of Variant): Variant;
  97. function VarArrayAsPSafeArray(const A: Variant): PVarArray;
  98. function VarArrayDimCount(const A: Variant) : LongInt;
  99. function VarArrayLowBound(const A: Variant; Dim : LongInt) : LongInt;
  100. function VarArrayHighBound(const A: Variant; Dim : LongInt) : LongInt;
  101. function VarArrayLock(const A: Variant): Pointer;
  102. procedure VarArrayUnlock(const A: Variant);
  103. function VarArrayRef(const A: Variant): Variant;
  104. function VarIsArray(const A: Variant): Boolean; inline;
  105. function VarIsArray(const A: Variant; AResolveByRef: Boolean): Boolean;
  106. function VarTypeIsValidArrayType(const aVarType: TVarType): Boolean;
  107. function VarTypeIsValidElementType(const aVarType: TVarType): Boolean;
  108. { Variant <--> Dynamic Arrays }
  109. procedure DynArrayToVariant(var V: Variant; const DynArray: Pointer; TypeInfo: Pointer);
  110. procedure DynArrayFromVariant(var DynArray: Pointer; const V: Variant; TypeInfo: Pointer);
  111. { Global constants }
  112. function Unassigned: Variant; // Unassigned standard constant
  113. function Null: Variant; // Null standard constant
  114. var
  115. EmptyParam: OleVariant;
  116. { Custom Variant base class }
  117. type
  118. TVarCompareResult = (crLessThan, crEqual, crGreaterThan);
  119. TCustomVariantType = class(TObject, IInterface)
  120. private
  121. FVarType: TVarType;
  122. protected
  123. function QueryInterface(const IID: TGUID; out Obj): HResult; virtual; stdcall;
  124. function _AddRef: Integer; stdcall;
  125. function _Release: Integer; stdcall;
  126. procedure SimplisticClear(var V: TVarData);
  127. procedure SimplisticCopy(var Dest: TVarData; const Source: TVarData; const Indirect: Boolean = False);
  128. procedure RaiseInvalidOp;
  129. procedure RaiseCastError;
  130. procedure RaiseDispError;
  131. function LeftPromotion(const V: TVarData; const Operation: TVarOp; out RequiredVarType: TVarType): Boolean; virtual;
  132. function RightPromotion(const V: TVarData; const Operation: TVarOp; out RequiredVarType: TVarType): Boolean; virtual;
  133. function OlePromotion(const V: TVarData; out RequiredVarType: TVarType): Boolean; virtual;
  134. procedure DispInvoke(Dest: PVarData; const Source: TVarData; CallDesc: PCallDesc; Params: Pointer); virtual;
  135. procedure VarDataInit(var Dest: TVarData);
  136. procedure VarDataClear(var Dest: TVarData);
  137. procedure VarDataCopy(var Dest: TVarData; const Source: TVarData);
  138. procedure VarDataCopyNoInd(var Dest: TVarData; const Source: TVarData);
  139. procedure VarDataCast(var Dest: TVarData; const Source: TVarData);
  140. procedure VarDataCastTo(var Dest: TVarData; const Source: TVarData; const aVarType: TVarType); overload;
  141. procedure VarDataCastTo(var Dest: TVarData; const aVarType: TVarType); overload;
  142. procedure VarDataCastToOleStr(var Dest: TVarData);
  143. procedure VarDataFromStr(var V: TVarData; const Value: string);
  144. procedure VarDataFromOleStr(var V: TVarData; const Value: WideString);
  145. function VarDataToStr(const V: TVarData): string;
  146. function VarDataIsEmptyParam(const V: TVarData): Boolean;
  147. function VarDataIsByRef(const V: TVarData): Boolean;
  148. function VarDataIsArray(const V: TVarData): Boolean;
  149. function VarDataIsOrdinal(const V: TVarData): Boolean;
  150. function VarDataIsFloat(const V: TVarData): Boolean;
  151. function VarDataIsNumeric(const V: TVarData): Boolean;
  152. function VarDataIsStr(const V: TVarData): Boolean;
  153. public
  154. constructor Create; overload;
  155. constructor Create(RequestedVarType: TVarType); overload;
  156. destructor Destroy; override;
  157. function IsClear(const V: TVarData): Boolean; virtual;
  158. procedure Cast(var Dest: TVarData; const Source: TVarData); virtual;
  159. procedure CastTo(var Dest: TVarData; const Source: TVarData; const aVarType: TVarType); virtual;
  160. procedure CastToOle(var Dest: TVarData; const Source: TVarData); virtual;
  161. procedure Clear(var V: TVarData); virtual; abstract;
  162. procedure Copy(var Dest: TVarData; const Source: TVarData; const Indirect: Boolean); virtual; abstract;
  163. procedure BinaryOp(var Left: TVarData; const Right: TVarData; const Operation: TVarOp); virtual;
  164. procedure UnaryOp(var Right: TVarData; const Operation: TVarOp); virtual;
  165. function CompareOp(const Left, Right: TVarData; const Operation: TVarOp): Boolean; virtual;
  166. procedure Compare(const Left, Right: TVarData; var Relationship: TVarCompareResult); virtual;
  167. property VarType: TVarType read FVarType;
  168. end;
  169. TCustomVariantTypeClass = class of TCustomVariantType;
  170. TVarDataArray = array of TVarData;
  171. IVarInvokeable = interface
  172. ['{1CB65C52-BBCB-41A6-9E58-7FB916BEEB2D}']
  173. function DoFunction(var Dest: TVarData; const V: TVarData;
  174. const Name: string; const Arguments: TVarDataArray): Boolean;
  175. function DoProcedure(const V: TVarData; const Name: string;
  176. const Arguments: TVarDataArray): Boolean;
  177. function GetProperty(var Dest: TVarData; const V: TVarData;
  178. const Name: string): Boolean;
  179. function SetProperty(const V: TVarData; const Name: string;
  180. const Value: TVarData): Boolean;
  181. end;
  182. TInvokeableVariantType = class(TCustomVariantType, IVarInvokeable)
  183. protected
  184. procedure DispInvoke(Dest: PVarData; const Source: TVarData;
  185. CallDesc: PCallDesc; Params: Pointer); override;
  186. public
  187. { IVarInvokeable }
  188. function DoFunction(var Dest: TVarData; const V: TVarData;
  189. const Name: string; const Arguments: TVarDataArray): Boolean; virtual;
  190. function DoProcedure(const V: TVarData; const Name: string;
  191. const Arguments: TVarDataArray): Boolean; virtual;
  192. function GetProperty(var Dest: TVarData; const V: TVarData;
  193. const Name: string): Boolean; virtual;
  194. function SetProperty(const V: TVarData; const Name: string;
  195. const Value: TVarData): Boolean; virtual;
  196. end;
  197. IVarInstanceReference = interface
  198. ['{5C176802-3F89-428D-850E-9F54F50C2293}']
  199. function GetInstance(const V: TVarData): TObject;
  200. end;
  201. TPublishableVariantType = class(TInvokeableVariantType, IVarInstanceReference)
  202. protected
  203. { IVarInstanceReference }
  204. function GetInstance(const V: TVarData): TObject; virtual; abstract;
  205. public
  206. function GetProperty(var Dest: TVarData; const V: TVarData;
  207. const Name: string): Boolean; override;
  208. function SetProperty(const V: TVarData; const Name: string;
  209. const Value: TVarData): Boolean; override;
  210. end;
  211. function FindCustomVariantType(const aVarType: TVarType;
  212. out CustomVariantType: TCustomVariantType): Boolean; overload;
  213. function FindCustomVariantType(const TypeName: string;
  214. out CustomVariantType: TCustomVariantType): Boolean; overload;
  215. type
  216. TAnyProc = procedure (var V: TVarData);
  217. TVarDispProc = procedure (Dest: PVariant; const Source: Variant;
  218. CallDesc: PCallDesc; Params: Pointer); cdecl;
  219. Const
  220. CMaxNumberOfCustomVarTypes = $0EFF;
  221. CMinVarType = $0100;
  222. CMaxVarType = CMinVarType + CMaxNumberOfCustomVarTypes;
  223. CIncVarType = $000F;
  224. CFirstUserType = CMinVarType + CIncVarType;
  225. var
  226. NullEqualityRule: TNullCompareRule = ncrLoose;
  227. NullMagnitudeRule: TNullCompareRule = ncrLoose;
  228. NullStrictConvert: Boolean = true;
  229. NullAsStringValue: string = '';
  230. PackVarCreation: Boolean = True;
  231. {$ifndef FPUNONE}
  232. OleVariantInt64AsDouble: Boolean = False;
  233. {$endif}
  234. VarDispProc: TVarDispProc;
  235. ClearAnyProc: TAnyProc; { Handler clearing a varAny }
  236. ChangeAnyProc: TAnyProc; { Handler to change any to Variant }
  237. RefAnyProc: TAnyProc; { Handler to add a reference to an varAny }
  238. InvalidCustomVariantType : TCustomVariantType;
  239. procedure VarCastError;
  240. procedure VarCastError(const ASourceType, ADestType: TVarType);
  241. procedure VarCastErrorOle(const ASourceType: TVarType);
  242. procedure VarInvalidOp;
  243. procedure VarInvalidOp(const aLeft, aRight: TVarType; aOpCode: TVarOp);
  244. procedure VarInvalidOp(const aRight: TVarType; aOpCode: TVarOp);
  245. procedure VarInvalidNullOp;
  246. procedure VarBadTypeError;
  247. procedure VarOverflowError;
  248. procedure VarOverflowError(const ASourceType, ADestType: TVarType);
  249. procedure VarBadIndexError;
  250. procedure VarArrayLockedError;
  251. procedure VarNotImplError;
  252. procedure VarOutOfMemoryError;
  253. procedure VarInvalidArgError;
  254. procedure VarInvalidArgError(AType: TVarType);
  255. procedure VarUnexpectedError;
  256. procedure VarRangeCheckError(const AType: TVarType);
  257. procedure VarRangeCheckError(const ASourceType, ADestType: TVarType);
  258. procedure VarArrayCreateError;
  259. procedure VarResultCheck(AResult: HRESULT);{$IFDEF VARIANTINLINE}inline;{$ENDIF VARIANTINLINE}
  260. procedure VarResultCheck(AResult: HRESULT; ASourceType, ADestType: TVarType);
  261. procedure HandleConversionException(const ASourceType, ADestType: TVarType);
  262. function VarTypeAsText(const AType: TVarType): string;
  263. function FindVarData(const V: Variant): PVarData;
  264. const
  265. VarOpAsText : array[TVarOp] of string = (
  266. '+', {opAdd}
  267. '-', {opSubtract}
  268. '*', {opMultiply}
  269. '/', {opDivide}
  270. 'div', {opIntDivide}
  271. 'mod', {opModulus}
  272. 'shl', {opShiftLeft}
  273. 'shr', {opShiftRight}
  274. 'and', {opAnd}
  275. 'or', {opOr}
  276. 'xor', {opXor}
  277. '', {opCompare}
  278. '-', {opNegate}
  279. 'not', {opNot}
  280. '=', {opCmpEq}
  281. '<>', {opCmpNe}
  282. '<', {opCmpLt}
  283. '<=', {opCmpLe}
  284. '>', {opCmpGt}
  285. '>=', {opCmpGe}
  286. '**' {opPower}
  287. );
  288. { Typinfo unit Variant routines have been moved here, so as not to make TypInfo dependent on variants }
  289. Function GetPropValue(Instance: TObject; const PropName: string): Variant;
  290. Function GetPropValue(Instance: TObject; const PropName: string; PreferStrings: Boolean): Variant;
  291. Procedure SetPropValue(Instance: TObject; const PropName: string; const Value: Variant);
  292. Function GetVariantProp(Instance: TObject; PropInfo : PPropInfo): Variant;
  293. Function GetVariantProp(Instance: TObject; const PropName: string): Variant;
  294. Procedure SetVariantProp(Instance: TObject; const PropName: string; const Value: Variant);
  295. Procedure SetVariantProp(Instance: TObject; PropInfo : PPropInfo; const Value: Variant);
  296. {$IFDEF DEBUG_VARIANTS}
  297. var
  298. __DEBUG_VARIANTS: Boolean = False;
  299. {$ENDIF}
  300. implementation
  301. uses
  302. Math,
  303. VarUtils;
  304. {$IFOPT R-} {$DEFINE RANGECHECKINGOFF} {$ENDIF}
  305. {$IFOPT Q-} {$DEFINE OVERFLOWCHECKINGOFF} {$ENDIF}
  306. var
  307. customvarianttypes : array of TCustomVariantType;
  308. customvarianttypelock : trtlcriticalsection;
  309. customvariantcurrtype : LongInt;
  310. const
  311. { all variants for which vType and varComplexType = 0 do not require
  312. finalization. }
  313. varComplexType = $BFE8;
  314. procedure DoVarClearComplex(var v : TVarData); forward;
  315. procedure DoVarCopy(var Dest : TVarData; const Source : TVarData); forward;
  316. procedure DoVarCast(var aDest : TVarData; const aSource : TVarData; aVarType : LongInt); forward;
  317. procedure DoVarClear(var v : TVarData); inline;
  318. begin
  319. if v.vType and varComplexType <> 0 then
  320. DoVarClearComplex(v)
  321. else
  322. v.vType := varEmpty;
  323. end;
  324. procedure DoVarClearIfComplex(var v : TVarData); inline;
  325. begin
  326. if v.vType and varComplexType <> 0 then
  327. DoVarClearComplex(v);
  328. end;
  329. function AlignToPtr(p : Pointer) : Pointer;inline;
  330. begin
  331. {$IFDEF FPC_REQUIRES_PROPER_ALIGNMENT}
  332. Result:=align(p,SizeOf(p));
  333. {$ELSE FPC_REQUIRES_PROPER_ALIGNMENT}
  334. Result:=p;
  335. {$ENDIF FPC_REQUIRES_PROPER_ALIGNMENT}
  336. end;
  337. { ---------------------------------------------------------------------
  338. String Messages
  339. ---------------------------------------------------------------------}
  340. ResourceString
  341. SErrVarIsEmpty = 'Variant is empty';
  342. SErrInvalidIntegerRange = 'Invalid Integer range: %d';
  343. { ---------------------------------------------------------------------
  344. Auxiliary routines
  345. ---------------------------------------------------------------------}
  346. Procedure VariantError (Const Msg : String); inline;
  347. begin
  348. Raise EVariantError.Create(Msg);
  349. end;
  350. Procedure NotSupported(Meth: String);
  351. begin
  352. Raise EVariantError.CreateFmt('Method %s not yet supported.',[Meth]);
  353. end;
  354. type
  355. TVariantArrayIterator = object
  356. Bounds : PVarArrayBoundArray;
  357. Coords : PVarArrayCoorArray;
  358. Dims : SizeInt;
  359. constructor Init(aDims: SizeInt; aBounds : PVarArrayBoundArray);
  360. destructor Done;
  361. function Next : Boolean;
  362. { returns true if the iterator reached the end of the variant array }
  363. function AtEnd: Boolean;
  364. end;
  365. {$r-}
  366. constructor TVariantArrayIterator.Init(aDims: SizeInt; aBounds : PVarArrayBoundArray);
  367. var
  368. i : sizeint;
  369. begin
  370. Dims := aDims;
  371. Bounds := aBounds;
  372. GetMem(Coords, SizeOf(SizeInt) * Dims);
  373. { initialize coordinate counter }
  374. for i:= 0 to Pred(Dims) do
  375. Coords^[i] := Bounds^[i].LowBound;
  376. end;
  377. function TVariantArrayIterator.Next: Boolean;
  378. var
  379. Finished : Boolean;
  380. procedure IncDim(Dim : SizeInt);
  381. begin
  382. if Finished then
  383. Exit;
  384. Inc(Coords^[Dim]);
  385. if Coords^[Dim] >= Bounds^[Dim].LowBound + Bounds^[Dim].ElementCount then begin
  386. Coords^[Dim]:=Bounds^[Dim].LowBound;
  387. if Dim > 0 then
  388. IncDim(Pred(Dim))
  389. else
  390. Finished := True;
  391. end;
  392. end;
  393. begin
  394. Finished := False;
  395. IncDim(Pred(Dims));
  396. Result := not Finished;
  397. end;
  398. function TVariantArrayIterator.AtEnd: Boolean;
  399. var
  400. i : sizeint;
  401. begin
  402. result:=true;
  403. for i:=0 to Pred(Dims) do
  404. if Coords^[i] < Bounds^[i].LowBound + Bounds^[i].ElementCount then
  405. begin
  406. result:=false;
  407. exit;
  408. end;
  409. end;
  410. {$ifndef RANGECHECKINGOFF}
  411. {$r+}
  412. {$endif}
  413. destructor TVariantArrayIterator.done;
  414. begin
  415. FreeMem(Coords);
  416. end;
  417. type
  418. tdynarraybounds = array of SizeInt;
  419. tdynarraycoords = tdynarraybounds;
  420. tdynarrayelesize = tdynarraybounds;
  421. tdynarraypositions = array of Pointer;
  422. tdynarrayiter = object
  423. Bounds : tdynarraybounds;
  424. Coords : tdynarraycoords;
  425. elesize : tdynarrayelesize;
  426. positions : tdynarraypositions;
  427. Dims : SizeInt;
  428. data : Pointer;
  429. constructor init(d : Pointer;p : pdynarraytypeinfo;_dims: SizeInt;b : tdynarraybounds);
  430. function next : Boolean;
  431. destructor done;
  432. end;
  433. constructor tdynarrayiter.init(d : Pointer;p : pdynarraytypeinfo;_dims: SizeInt;b : tdynarraybounds);
  434. var
  435. i : sizeint;
  436. begin
  437. Bounds:=b;
  438. Dims:=_dims;
  439. SetLength(Coords,Dims);
  440. SetLength(elesize,Dims);
  441. SetLength(positions,Dims);
  442. positions[0]:=d;
  443. { initialize coordinate counter and elesize }
  444. for i:=0 to Dims-1 do
  445. begin
  446. Coords[i]:=0;
  447. if i>0 then
  448. positions[i]:=Pointer(positions[i-1]^);
  449. { skip kind and name }
  450. inc(Pointer(p),ord(pdynarraytypeinfo(p)^.namelen)+2);
  451. p:=AlignToPtr(p);
  452. elesize[i]:=psizeint(p)^;
  453. { skip elesize }
  454. inc(Pointer(p),SizeOf(sizeint));
  455. p:=pdynarraytypeinfo(ppointer(p)^);
  456. end;
  457. data:=positions[Dims-1];
  458. end;
  459. function tdynarrayiter.next : Boolean;
  460. var
  461. Finished : Boolean;
  462. procedure incdim(d : SizeInt);
  463. begin
  464. if Finished then
  465. exit;
  466. inc(Coords[d]);
  467. inc(Pointer(positions[d]),elesize[d]);
  468. if Coords[d]>=Bounds[d] then
  469. begin
  470. Coords[d]:=0;
  471. if d>0 then
  472. begin
  473. incdim(d-1);
  474. positions[d]:=Pointer(positions[d-1]^);
  475. end
  476. else
  477. Finished:=true;
  478. end;
  479. end;
  480. begin
  481. Finished:=False;
  482. incdim(Dims-1);
  483. data:=positions[Dims-1];
  484. Result:=not(Finished);
  485. end;
  486. destructor tdynarrayiter.done;
  487. begin
  488. Bounds:=nil;
  489. Coords:=nil;
  490. elesize:=nil;
  491. positions:=nil;
  492. end;
  493. { ---------------------------------------------------------------------
  494. VariantManager support
  495. ---------------------------------------------------------------------}
  496. procedure sysvarinit(var v : Variant);
  497. begin
  498. TVarData(V).vType := varEmpty;
  499. end;
  500. procedure sysvarclear(var v : Variant);
  501. begin
  502. if TVarData(v).vType and varComplexType <> 0 then
  503. VarClearProc(TVarData(V))
  504. else
  505. TVarData(v).vType := varEmpty;
  506. end;
  507. function Sysvartoint (const v : Variant) : Integer;
  508. begin
  509. if VarType(v) = varNull then
  510. if NullStrictConvert then
  511. VarCastError(varNull, varInt64)
  512. else
  513. Result := 0
  514. else
  515. Result := VariantToLongInt(TVarData(V));
  516. end;
  517. function Sysvartoint64 (const v : Variant) : Int64;
  518. begin
  519. if VarType(v) = varNull then
  520. if NullStrictConvert then
  521. VarCastError(varNull, varInt64)
  522. else
  523. Result := 0
  524. else
  525. Result := VariantToInt64(TVarData(V));
  526. end;
  527. function sysvartoword64 (const v : Variant) : QWord;
  528. begin
  529. if VarType(v) = varNull then
  530. if NullStrictConvert then
  531. VarCastError(varNull, varQWord)
  532. else
  533. Result := 0
  534. else
  535. Result := VariantToQWord (TVarData(V));
  536. end;
  537. function sysvartobool (const v : Variant) : Boolean;
  538. begin
  539. if VarType(v) = varNull then
  540. if NullStrictConvert then
  541. VarCastError(varNull, varBoolean)
  542. else
  543. Result := False
  544. else
  545. Result := VariantToBoolean(TVarData(V));
  546. end;
  547. {$ifndef FPUNONE}
  548. function sysvartoreal (const v : Variant) : Extended;
  549. var Handler: TCustomVariantType;
  550. dest: TVarData;
  551. begin
  552. if VarType(v) = varNull then
  553. if NullStrictConvert then
  554. VarCastError(varNull, varDouble)
  555. else
  556. Result := 0
  557. else if FindCustomVariantType(TVarData(v).vType, Handler) then
  558. begin
  559. VariantInit(dest);
  560. Handler.CastTo(dest, TVarData(v), varDouble);
  561. Result := dest.vDouble;
  562. end
  563. else
  564. Result := VariantToDouble(TVarData(V));
  565. end;
  566. {$endif}
  567. function sysvartocurr (const v : Variant) : Currency;
  568. begin
  569. if VarType(v) = varNull then
  570. if NullStrictConvert then
  571. VarCastError(varNull, varCurrency)
  572. else
  573. Result := 0
  574. else
  575. Result := VariantToCurrency(TVarData(V));
  576. end;
  577. procedure sysvartolstr (var s : AnsiString; const v : Variant);
  578. begin
  579. if VarType(v) = varNull then
  580. if NullStrictConvert then
  581. VarCastError(varNull, varString)
  582. else
  583. s := NullAsStringValue
  584. else
  585. S := VariantToAnsiString(TVarData(V));
  586. end;
  587. procedure sysvartopstr (var s; const v : Variant);
  588. begin
  589. if VarType(v) = varNull then
  590. if NullStrictConvert then
  591. VarCastError(varNull, varString)
  592. else
  593. ShortString(s) := NullAsStringValue
  594. else
  595. ShortString(s) := VariantToShortString(TVarData(V));
  596. end;
  597. procedure sysvartowstr (var s : WideString; const v : Variant);
  598. begin
  599. if VarType(v) = varNull then
  600. if NullStrictConvert then
  601. VarCastError(varNull, varOleStr)
  602. else
  603. s := NullAsStringValue
  604. else
  605. S := VariantToWideString(TVarData(V));
  606. end;
  607. procedure sysvartointf (var Intf : IInterface; const v : Variant);
  608. begin
  609. case TVarData(v).vType of
  610. varEmpty:
  611. Intf := nil;
  612. varNull:
  613. if NullStrictConvert then
  614. VarCastError(varNull, varUnknown)
  615. else
  616. Intf := nil;
  617. varUnknown:
  618. Intf := IInterface(TVarData(v).vUnknown);
  619. varUnknown or varByRef:
  620. Intf := IInterface(TVarData(v).vPointer^);
  621. varDispatch:
  622. Intf := IInterface(TVarData(v).vDispatch);
  623. varDispatch or varByRef:
  624. Intf := IInterface(TVarData(v).vPointer^);
  625. varVariant, varVariant or varByRef: begin
  626. if not Assigned(TVarData(v).vPointer) then
  627. VarBadTypeError;
  628. sysvartointf(Intf, Variant(PVarData(TVarData(v).vPointer)^) );
  629. end;
  630. else
  631. VarCastError(TVarData(v).vType, varUnknown);
  632. end;
  633. end;
  634. procedure sysvartodisp (var Disp : IDispatch; const v : Variant);
  635. begin
  636. case TVarData(v).vType of
  637. varEmpty:
  638. Disp := nil;
  639. varNull:
  640. if NullStrictConvert then
  641. VarCastError(varNull, varDispatch)
  642. else
  643. Disp := nil;
  644. varUnknown:
  645. if IInterface(TVarData(v).vUnknown).QueryInterface(IDispatch, Disp) <> S_OK then
  646. VarCastError(varUnknown, varDispatch);
  647. varUnknown or varByRef:
  648. if IInterface(TVarData(v).vPointer^).QueryInterface(IDispatch, Disp) <> S_OK then
  649. VarCastError(varUnknown or varByRef, varDispatch);
  650. varDispatch:
  651. Disp := IDispatch(TVarData(v).vDispatch);
  652. varDispatch or varByRef:
  653. Disp := IDispatch(TVarData(v).vPointer^);
  654. varVariant, varVariant or varByRef: begin
  655. if not Assigned(TVarData(v).vPointer) then
  656. VarBadTypeError;
  657. sysvartodisp(Disp, Variant(PVarData(TVarData(v).vPointer)^) );
  658. end;
  659. else
  660. VarCastError(TVarData(v).vType, varDispatch);
  661. end;
  662. end;
  663. {$ifndef FPUNONE}
  664. function sysvartotdatetime (const v : Variant) : TDateTime;
  665. begin
  666. if VarType(v) = varNull then
  667. if NullStrictConvert then
  668. VarCastError(varNull, varDate)
  669. else
  670. Result := 0
  671. else
  672. Result:=VariantToDate(TVarData(v));
  673. end;
  674. {$endif}
  675. function DynamicArrayIsRectangular(p : Pointer;TypeInfo : Pointer) : Boolean;
  676. var
  677. arraysize,i : sizeint;
  678. begin
  679. Result := False;
  680. { get TypeInfo of second level }
  681. { skip kind and name }
  682. inc(Pointer(TypeInfo),ord(pdynarraytypeinfo(TypeInfo)^.namelen)+2);
  683. TypeInfo:=AlignToPtr(TypeInfo);
  684. TypeInfo:=ppointer(TypeInfo+SizeOf(sizeint))^;
  685. { check recursively? }
  686. if assigned(pdynarraytypeinfo(TypeInfo)) and (pdynarraytypeinfo(TypeInfo)^.kind=byte(tkDynArray)) then
  687. begin
  688. { set to dimension of first element }
  689. arraysize:=psizeint(ppointer(p)^-SizeOf(sizeint))^;
  690. { walk through all elements }
  691. for i:=1 to psizeint(p-SizeOf(sizeint))^ do
  692. begin
  693. { ... and check dimension }
  694. if psizeint(ppointer(p)^-SizeOf(sizeint))^<>arraysize then
  695. exit;
  696. if not(DynamicArrayIsRectangular(ppointer(p)^,TypeInfo)) then
  697. exit;
  698. inc(p,SizeOf(Pointer));
  699. end;
  700. end;
  701. Result:=true;
  702. end;
  703. procedure sysvartodynarray (var dynarr : Pointer; const v : Variant; TypeInfo : Pointer);
  704. begin
  705. DynArrayFromVariant(dynarr, v, TypeInfo);
  706. end;
  707. procedure sysvarfrombool (var Dest : Variant; const Source : Boolean);
  708. begin
  709. DoVarClearIfComplex(TVarData(Dest));
  710. with TVarData(Dest) do begin
  711. vType := varBoolean;
  712. vBoolean := Source;
  713. end;
  714. end;
  715. procedure VariantErrorInvalidIntegerRange(Range: LongInt);
  716. begin
  717. VariantError(Format(SErrInvalidIntegerRange,[Range]));
  718. end;
  719. procedure sysvarfromint (var Dest : Variant; const Source, Range : LongInt);
  720. begin
  721. DoVarClearIfComplex(TVarData(Dest));
  722. with TVarData(Dest) do
  723. if PackVarCreation then
  724. case Range of
  725. -4 : begin
  726. vType := varInteger;
  727. vInteger := Source;
  728. end;
  729. -2 : begin
  730. vType := varSmallInt;
  731. vSmallInt := Source;
  732. end;
  733. -1 : Begin
  734. vType := varShortInt;
  735. vshortint := Source;
  736. end;
  737. 1 : begin
  738. vType := varByte;
  739. vByte := Source;
  740. end;
  741. 2 : begin
  742. vType := varWord;
  743. vWord := Source;
  744. end;
  745. 4 : Begin
  746. vType := varLongWord;
  747. {use vInteger, not vLongWord as the value came passed in as an Integer }
  748. vInteger := Source;
  749. end;
  750. else
  751. VariantErrorInvalidIntegerRange(Range);
  752. end
  753. else begin
  754. vType := varInteger;
  755. vInteger := Source;
  756. end;
  757. end;
  758. procedure sysvarfromint64 (var Dest : Variant; const Source : Int64);
  759. begin
  760. DoVarClearIfComplex(TVarData(Dest));
  761. with TVarData(Dest) do begin
  762. vType := varInt64;
  763. vInt64 := Source;
  764. end;
  765. end;
  766. procedure sysvarfromword64 (var Dest : Variant; const Source : QWord);
  767. begin
  768. DoVarClearIfComplex(TVarData(Dest));
  769. with TVarData(Dest) do begin
  770. vType := varQWord;
  771. vQWord := Source;
  772. end;
  773. end;
  774. {$ifndef FPUNONE}
  775. procedure sysvarfromreal (var Dest : Variant; const Source : Extended);
  776. begin
  777. DoVarClearIfComplex(TVarData(Dest));
  778. with TVarData(Dest) do begin
  779. vType := varDouble;
  780. vDouble := Source;
  781. end;
  782. end;
  783. procedure sysvarfromsingle (var Dest : Variant; const Source : single);
  784. begin
  785. DoVarClearIfComplex(TVarData(Dest));
  786. with TVarData(Dest) do begin
  787. vType := varSingle;
  788. vSingle := Source;
  789. end;
  790. end;
  791. procedure sysvarfromdouble (var Dest : Variant; const Source : double);
  792. begin
  793. DoVarClearIfComplex(TVarData(Dest));
  794. with TVarData(Dest) do begin
  795. vType := varDouble;
  796. vDouble := Source;
  797. end;
  798. end;
  799. {$endif}
  800. procedure sysvarfromcurr (var Dest : Variant; const Source : Currency);
  801. begin
  802. DoVarClearIfComplex(TVarData(Dest));
  803. with TVarData(Dest) do begin
  804. vType := varCurrency;
  805. vCurrency := Source;
  806. end;
  807. end;
  808. {$ifndef FPUNONE}
  809. procedure sysvarfromtdatetime (var Dest : Variant; const Source : TDateTime);
  810. begin
  811. DoVarClearIfComplex(TVarData(Dest));
  812. with TVarData(Dest) do begin
  813. vType := varDate;
  814. vDate := Source;
  815. end;
  816. end;
  817. {$endif}
  818. procedure sysvarfrompstr (var Dest : Variant; const Source : ShortString);
  819. begin
  820. DoVarClearIfComplex(TVarData(Dest));
  821. with TVarData(Dest) do begin
  822. vType := varString;
  823. vString := nil;
  824. AnsiString(vString) := Source;
  825. end;
  826. end;
  827. procedure sysvarfromlstr (var Dest : Variant; const Source : AnsiString);
  828. begin
  829. DoVarClearIfComplex(TVarData(Dest));
  830. with TVarData(Dest) do begin
  831. vType := varString;
  832. vString := nil;
  833. AnsiString(vString) := Source;
  834. end;
  835. end;
  836. procedure sysvarfromwstr (var Dest : Variant; const Source : WideString);
  837. begin
  838. DoVarClearIfComplex(TVarData(Dest));
  839. with TVarData(Dest) do begin
  840. vType := varOleStr;
  841. vOleStr := nil;
  842. WideString(Pointer(vOleStr)) := Source;
  843. end;
  844. end;
  845. procedure sysvarfromintf(var Dest : Variant; const Source : IInterface);
  846. begin
  847. DoVarClearIfComplex(TVarData(Dest));
  848. with TVarData(Dest) do begin
  849. vUnknown := nil;
  850. IInterface(vUnknown) := Source;
  851. vType := varUnknown;
  852. end;
  853. end;
  854. procedure sysvarfromdisp(var Dest : Variant; const Source : IDispatch);
  855. begin
  856. DoVarClearIfComplex(TVarData(Dest));
  857. with TVarData(Dest) do begin
  858. vUnknown := nil;
  859. IDispatch(vDispatch) := Source;
  860. vType := varDispatch;
  861. end;
  862. end;
  863. type
  864. TCommonType = (ctEmpty,ctAny,ctError,ctLongInt,ctBoolean,
  865. {$ifndef FPUNONE}
  866. ctFloat,ctDate,ctCurrency,
  867. {$endif}
  868. ctInt64,ctNull,ctWideStr,ctString);
  869. TCommonVarType = varEmpty..varQWord;
  870. const
  871. {$ifdef FPUNONE}
  872. ctFloat = ctError;
  873. ctDate = ctError;
  874. ctCurrency = ctError;
  875. {$endif}
  876. { get the basic type for a Variant type }
  877. VarTypeToCommonType : array[TCommonVarType] of TCommonType =
  878. (ctEmpty, // varEmpty = 0;
  879. ctNull, // varNull = 1;
  880. ctLongInt, // varSmallInt = 2;
  881. ctLongInt, // varInteger = 3;
  882. ctFloat, // varSingle = 4;
  883. ctFloat, // varDouble = 5;
  884. ctCurrency, // varCurrency = 6;
  885. ctDate, // varDate = 7;
  886. ctWideStr, // varOleStr = 8;
  887. ctError, // varDispatch = 9;
  888. ctError, // varError = 10;
  889. ctBoolean, // varBoolean = 11;
  890. ctError, // varVariant = 12;
  891. ctError, // varUnknown = 13;
  892. ctError, // ??? 15
  893. ctError, // varDecimal = 14;
  894. ctLongInt, // varShortInt = 16;
  895. ctLongInt, // varByte = 17;
  896. ctLongInt, // varWord = 18;
  897. ctInt64, // varLongWord = 19;
  898. ctInt64, // varInt64 = 20;
  899. ctInt64 // varQWord = 21;
  900. );
  901. { map a basic type back to a Variant type }
  902. { Not used yet
  903. CommonTypeToVarType : array[TCommonType] of TVarType =
  904. (
  905. varEmpty,
  906. varany,
  907. varError,
  908. varInteger,
  909. varDouble,
  910. varBoolean,
  911. varInt64,
  912. varNull,
  913. varOleStr,
  914. varDate,
  915. varCurrency,
  916. varString
  917. );
  918. }
  919. function MapToCommonType(const vType : TVarType) : TCommonType;
  920. begin
  921. case vType of
  922. Low(TCommonVarType)..High(TCommonVarType):
  923. Result := VarTypeToCommonType[vType];
  924. varString:
  925. Result:=ctString;
  926. varAny:
  927. Result:=ctAny;
  928. else
  929. Result:=ctError;
  930. end;
  931. end;
  932. const
  933. FindCmpCommonType : array[TCommonType, TCommonType] of TCommonType = (
  934. { ctEmpty ctAny ctError ctLongInt ctBoolean ctFloat ctDate ctCurrency ctInt64 ctNull ctWideStr ctString }
  935. ({ ctEmpty } ctEmpty, ctEmpty, ctError, ctEmpty, ctEmpty, {$ifndef FPUNONE}ctEmpty, ctEmpty, ctEmpty, {$endif}ctEmpty, ctEmpty, ctEmpty, ctEmpty ),
  936. ({ ctAny } ctEmpty, ctAny, ctError, ctAny, ctAny, {$ifndef FPUNONE}ctAny, ctAny, ctAny, {$endif}ctAny, ctAny, ctAny, ctAny ),
  937. ({ ctError } ctError, ctError, ctError, ctError, ctError, {$ifndef FPUNONE}ctError, ctError, ctError, {$endif}ctError, ctError, ctError, ctError ),
  938. ({ ctLongInt } ctEmpty, ctAny, ctError, ctLongInt, ctBoolean, {$ifndef FPUNONE}ctFloat, ctDate, ctCurrency, {$endif}ctInt64, ctNull, ctFloat, ctFloat ),
  939. ({ ctBoolean } ctEmpty, ctAny, ctError, ctLongInt, ctBoolean, {$ifndef FPUNONE}ctFloat, ctDate, ctCurrency, {$endif}ctInt64, ctNull, ctWideStr, ctString ),
  940. {$ifndef FPUNONE}
  941. ({ ctFloat } ctEmpty, ctAny, ctError, ctFloat, ctFloat, ctFloat, ctDate, ctCurrency, ctFloat, ctNull, ctFloat, ctFloat ),
  942. ({ ctDate } ctEmpty, ctAny, ctError, ctDate, ctDate, ctDate, ctDate, ctDate, ctDate, ctNull, ctDate, ctDate ),
  943. ({ ctCurrency } ctEmpty, ctAny, ctError, ctCurrency, ctCurrency, ctCurrency,ctDate, ctCurrency, ctCurrency, ctNull, ctCurrency, ctCurrency ),
  944. {$endif}
  945. ({ ctInt64 } ctEmpty, ctAny, ctError, ctInt64, ctInt64, {$ifndef FPUNONE}ctFloat, ctDate, ctCurrency, {$endif}ctInt64, ctNull, ctFloat, ctFloat ),
  946. ({ ctNull } ctEmpty, ctAny, ctError, ctNull, ctNull, {$ifndef FPUNONE}ctNull, ctNull, ctNull, {$endif}ctNull, ctNull, ctNull, ctNull ),
  947. ({ ctWideStr } ctEmpty, ctAny, ctError, ctFloat, ctWideStr, {$ifndef FPUNONE}ctFloat, ctDate, ctCurrency, {$endif}ctFloat, ctNull, ctWideStr, ctWideStr ),
  948. ({ ctString } ctEmpty, ctAny, ctError, ctFloat, ctString, {$ifndef FPUNONE}ctFloat, ctDate, ctCurrency, {$endif}ctFloat, ctNull, ctWideStr, ctString )
  949. );
  950. function DoVarCmpSimple (const Left, Right, Common: TCommonType) : ShortInt; inline;
  951. begin
  952. if Left = Common then
  953. if Right = Common then
  954. Result := 0
  955. else
  956. Result := -1
  957. else
  958. Result := 1;
  959. end;
  960. function DoVarCmpAny(const Left, Right: TVarData; const OpCode: TVarOp) : ShortInt;
  961. begin
  962. VarInvalidOp(Left.vType, Right.vType, OpCode);
  963. Result:=0;
  964. end;
  965. function DoVarCmpLongInt(const Left, Right: LongInt): ShortInt; inline;
  966. begin
  967. if Left < Right then
  968. Result := -1
  969. else if Left > Right then
  970. Result := 1
  971. else
  972. Result := 0;
  973. end;
  974. {$ifndef FPUNONE}
  975. function DoVarCmpFloat(const Left, Right: Double; const OpCode: TVarOp): ShortInt;
  976. begin
  977. if Left = Right then
  978. Result := 0
  979. else if (OpCode in [opCmpEq, opCmpNe]) or (Left < Right) then
  980. Result := -1
  981. else
  982. Result := 1;
  983. end;
  984. {$endif}
  985. function DoVarCmpInt64(const Left, Right: Int64): ShortInt;
  986. begin
  987. if Left < Right then
  988. Result := -1
  989. else if Left > Right then
  990. Result := 1
  991. else
  992. Result := 0;
  993. end;
  994. function DoVarCmpNull(const Left, Right: TCommonType; const OpCode: TVarOp) : ShortInt;
  995. const
  996. ResultMap: array [Boolean, opCmpEq..opCmpGe] of ShortInt =
  997. ( ( -1, 0, 0, 1, 0, -1 ), ( 0, -1, -1, -1, 1, 1 ) );
  998. begin
  999. if OpCode in [opCmpEq, opCmpNe] then
  1000. case NullEqualityRule of
  1001. ncrError: VarInvalidNullOp;
  1002. ncrStrict: Result := ResultMap[False, OpCode];
  1003. ncrLoose: Result := ResultMap[(Left = Right) xor (OpCode = opCmpNe), OpCode];
  1004. end
  1005. else
  1006. case NullMagnitudeRule of
  1007. ncrError: VarInvalidNullOp;
  1008. ncrStrict: Result := ResultMap[False, OpCode];
  1009. ncrLoose: Result := DoVarCmpSimple(Left, Right, ctNull);
  1010. end;
  1011. end;
  1012. function DoVarCmpCurr(const Left, Right: Currency): ShortInt;
  1013. begin
  1014. if Left < Right then
  1015. Result := -1
  1016. else if Left > Right then
  1017. Result := 1
  1018. else
  1019. Result := 0;
  1020. end;
  1021. function DoVarCmpWStrDirect(const Left, Right: Pointer; const OpCode: TVarOp): ShortInt; inline;
  1022. begin
  1023. { we can do this without ever copying the string }
  1024. if OpCode in [opCmpEq, opCmpNe] then
  1025. if Length(WideString(Left)) <> Length(WideString(Right)) then
  1026. Exit(-1);
  1027. Result := WideCompareStr(
  1028. WideString(Left),
  1029. WideString(Right)
  1030. );
  1031. end;
  1032. function DoVarCmpWStr(const Left, Right: TVarData; const OpCode: TVarOp): ShortInt;
  1033. begin
  1034. { keep the temps away from the main proc }
  1035. Result := DoVarCmpWStrDirect(Pointer(VariantToWideString(Left)),
  1036. Pointer(VariantToWideString(Right)), OpCode);
  1037. end;
  1038. function DoVarCmpLStrDirect(const Left, Right: Pointer; const OpCode: TVarOp): ShortInt; inline;
  1039. begin
  1040. { we can do this without ever copying the string }
  1041. if OpCode in [opCmpEq, opCmpNe] then
  1042. if Length(AnsiString(Left)) <> Length(AnsiString(Right)) then
  1043. Exit(-1);
  1044. Result := CompareStr(
  1045. AnsiString(Left),
  1046. AnsiString(Right)
  1047. );
  1048. end;
  1049. function DoVarCmpLStr(const Left, Right: TVarData; const OpCode: TVarOp): ShortInt;
  1050. begin
  1051. { keep the temps away from the main proc }
  1052. Result := DoVarCmpLStrDirect(Pointer(VariantToAnsiString(Left)),
  1053. Pointer(VariantToAnsiString(Right)), OpCode);
  1054. end;
  1055. function DoVarCmpComplex(const Left, Right: TVarData; const OpCode: TVarOp): ShortInt;
  1056. var Handler: TCustomVariantType;
  1057. CmpRes: boolean;
  1058. begin
  1059. if FindCustomVariantType(Left.vType, Handler) then
  1060. CmpRes := Handler.CompareOp(Left, Right, OpCode)
  1061. else if FindCustomVariantType(Right.vType, Handler) then
  1062. CmpRes := Handler.CompareOp(Left, Right, OpCode)
  1063. else
  1064. VarInvalidOp(Left.vType, Right.vType, OpCode);
  1065. case OpCode of
  1066. opCmpEq:
  1067. if CmpRes then
  1068. Result:=0
  1069. else
  1070. Result:=1;
  1071. opCmpNe:
  1072. if CmpRes then
  1073. Result:=1
  1074. else
  1075. Result:=0;
  1076. opCmpLt,
  1077. opCmpLe:
  1078. if CmpRes then
  1079. Result:=-1
  1080. else
  1081. Result:=1;
  1082. opCmpGt,
  1083. opCmpGe:
  1084. if CmpRes then
  1085. Result:=1
  1086. else
  1087. Result:=-1;
  1088. end;
  1089. end;
  1090. function DoVarCmp(const vl, vr : TVarData; const OpCode : TVarOp) : ShortInt;
  1091. var
  1092. lct: TCommonType;
  1093. rct: TCommonType;
  1094. begin
  1095. { as the function in cvarutil.inc can handle varByRef correctly we simply
  1096. resolve the final type }
  1097. lct := MapToCommonType(VarTypeDeRef(vl));
  1098. rct := MapToCommonType(VarTypeDeRef(vr));
  1099. {$IFDEF DEBUG_VARIANTS}
  1100. if __DEBUG_VARIANTS then begin
  1101. WriteLn('DoVarCmp $', IntToHex(Cardinal(@vl),8), ' ', GetEnumName(TypeInfo(TVarOp), Ord(OpCode)) ,' $', IntToHex(Cardinal(@vr),8));
  1102. DumpVariant('DoVarCmp/vl', vl);
  1103. WriteLn('lct ', GetEnumName(TypeInfo(TCommonType), Ord(lct)));
  1104. DumpVariant('DoVarCmp/vr', vr);
  1105. WriteLn('rct ', GetEnumName(TypeInfo(TCommonType), Ord(rct)));
  1106. WriteLn('common ', GetEnumName(TypeInfo(TCommonType), Ord(FindCmpCommonType[lct, rct])));
  1107. end;
  1108. {$ENDIF}
  1109. case FindCmpCommonType[lct, rct] of
  1110. ctEmpty: Result := DoVarCmpSimple(lct, rct, ctEmpty);
  1111. ctAny: Result := DoVarCmpAny(vl, vr, OpCode);
  1112. ctLongInt: Result := DoVarCmpLongInt(VariantToLongInt(vl), VariantToLongInt(vr));
  1113. {$ifndef FPUNONE}
  1114. ctFloat: Result := DoVarCmpFloat(VariantToDouble(vl), VariantToDouble(vr), OpCode);
  1115. {$endif}
  1116. ctBoolean: Result := DoVarCmpLongInt(LongInt(VariantToBoolean(vl)), LongInt(VariantToBoolean(vr)));
  1117. ctInt64: Result := DoVarCmpInt64(VariantToInt64(vl), VariantToInt64(vr));
  1118. ctNull: Result := DoVarCmpNull(lct, rct, OpCode);
  1119. ctWideStr:
  1120. if (vl.vType = varOleStr) and (vr.vType = varOleStr) then
  1121. Result := DoVarCmpWStrDirect(Pointer(vl.vOleStr), Pointer(vr.vOleStr), OpCode)
  1122. else
  1123. Result := DoVarCmpWStr(vl, vr, OpCode);
  1124. {$ifndef FPUNONE}
  1125. ctDate: Result := DoVarCmpFloat(VariantToDate(vl), VariantToDate(vr), OpCode);
  1126. ctCurrency: Result := DoVarCmpCurr(VariantToCurrency(vl), VariantToCurrency(vr));
  1127. {$endif}
  1128. ctString:
  1129. if (vl.vType = varString) and (vr.vType = varString) then
  1130. Result := DoVarCmpLStrDirect(Pointer(vl.vString), Pointer(vr.vString), OpCode)
  1131. else
  1132. Result := DoVarCmpLStr(vl, vr, OpCode);
  1133. else
  1134. Result := DoVarCmpComplex(vl, vr, OpCode);
  1135. end;
  1136. end;
  1137. function syscmpop (const Left, Right : Variant; const OpCode : TVarOp) : Boolean;
  1138. var
  1139. CmpRes : ShortInt;
  1140. begin
  1141. CmpRes:=DoVarCmp(TVarData(Left),TVarData(Right),OpCode);
  1142. case OpCode of
  1143. opCmpEq:
  1144. Result:=CmpRes=0;
  1145. opCmpNe:
  1146. Result:=CmpRes<>0;
  1147. opCmpLt:
  1148. Result:=CmpRes<0;
  1149. opCmpLe:
  1150. Result:=CmpRes<=0;
  1151. opCmpGt:
  1152. Result:=CmpRes>0;
  1153. opCmpGe:
  1154. Result:=CmpRes>=0;
  1155. else
  1156. VarInvalidOp;
  1157. end;
  1158. end;
  1159. const
  1160. FindOpCommonType : array[TCommonType,TCommonType] of TCommonType = (
  1161. { ctEmpty ctAny ctError ctLongInt ctBoolean ctFloat ctDate ctCurrency ctInt64 ctNull ctWideStr ctString }
  1162. ({ ctEmpty } ctEmpty, ctAny, ctError, ctEmpty, ctEmpty, {$ifndef FPUNONE}ctEmpty, ctEmpty, ctEmpty, {$endif}ctEmpty, ctEmpty, ctEmpty, ctEmpty ),
  1163. ({ ctAny } ctAny, ctAny, ctError, ctAny, ctAny, {$ifndef FPUNONE}ctAny, ctAny, ctAny, {$endif}ctAny, ctAny, ctAny, ctAny ),
  1164. ({ ctError } ctError, ctError, ctError, ctError, ctError, {$ifndef FPUNONE}ctError, ctError, ctError, {$endif}ctError, ctError, ctError, ctError ),
  1165. ({ ctLongInt } ctEmpty, ctAny, ctError, ctLongInt, ctBoolean, {$ifndef FPUNONE}ctFloat, ctDate, ctCurrency, {$endif}ctInt64, ctNull, ctFloat, ctFloat ),
  1166. ({ ctBoolean } ctEmpty, ctAny, ctError, ctLongInt, ctBoolean, {$ifndef FPUNONE}ctFloat, ctDate, ctCurrency, {$endif}ctInt64, ctNull, ctBoolean, ctBoolean ),
  1167. {$ifndef FPUNONE}
  1168. ({ ctFloat } ctEmpty, ctAny, ctError, ctFloat, ctFloat, ctFloat, ctDate, ctCurrency, ctFloat, ctNull, ctFloat, ctFloat ),
  1169. ({ ctDate } ctEmpty, ctAny, ctError, ctDate, ctDate, ctDate, ctDate, ctDate, ctDate, ctNull, ctDate, ctDate ),
  1170. ({ ctCurrency } ctEmpty, ctAny, ctError, ctCurrency, ctCurrency, ctCurrency, ctDate, ctCurrency, ctCurrency, ctNull, ctCurrency, ctCurrency ),
  1171. {$endif}
  1172. ({ ctInt64 } ctEmpty, ctAny, ctError, ctInt64, ctInt64, {$ifndef FPUNONE}ctFloat, ctDate, ctCurrency, {$endif}ctInt64, ctNull, ctFloat, ctFloat ),
  1173. ({ ctNull } ctEmpty, ctAny, ctError, ctNull, ctNull, {$ifndef FPUNONE}ctNull, ctNull, ctNull, {$endif}ctNull, ctNull, ctNull, ctNull ),
  1174. ({ ctWideStr } ctEmpty, ctAny, ctError, ctFloat, ctBoolean, {$ifndef FPUNONE}ctFloat, ctDate, ctCurrency, {$endif}ctFloat, ctNull, ctWideStr, ctWideStr ),
  1175. ({ ctString } ctEmpty, ctAny, ctError, ctFloat, ctBoolean, {$ifndef FPUNONE}ctFloat, ctDate, ctCurrency, {$endif}ctFloat, ctNull, ctWideStr, ctString )
  1176. );
  1177. procedure DoVarOpFloat(var vl :TVarData; const vr : TVarData; const OpCode : TVarOp);
  1178. {$ifndef FPUNONE}
  1179. var
  1180. l, r : Double;
  1181. begin
  1182. l := VariantToDouble(vl);
  1183. r := VariantToDouble(vr);
  1184. case OpCode of
  1185. opAdd : l := l + r;
  1186. opSubtract : l := l - r;
  1187. opMultiply : l := l * r;
  1188. opDivide : l := l / r;
  1189. opPower : l := l ** r;
  1190. else
  1191. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1192. end;
  1193. DoVarClearIfComplex(vl);
  1194. vl.vType := varDouble;
  1195. vl.vDouble := l;
  1196. {$else}
  1197. begin
  1198. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1199. {$endif}
  1200. end;
  1201. procedure DoVarOpAny(var vl : TVarData; const vr : TVarData; const OpCode : TVarOp);
  1202. begin
  1203. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1204. end;
  1205. procedure DoVarOpLongInt(var vl : TVarData; const vr : TVarData; const OpCode : TVarOp);
  1206. var
  1207. l, r: LongInt;
  1208. begin
  1209. l := VariantToLongint(vl);
  1210. r := VariantToLongint(vr);
  1211. case OpCode of
  1212. opIntDivide : l := l div r;
  1213. opModulus : l := l mod r;
  1214. opShiftLeft : l := l shl r;
  1215. opShiftRight : l := l shr r;
  1216. opAnd : l := l and r;
  1217. opOr : l := l or r;
  1218. opXor : l := l xor r;
  1219. else
  1220. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1221. end;
  1222. DoVarClearIfComplex(vl);
  1223. vl.vType := varInteger;
  1224. vl.vInteger := l;
  1225. end;
  1226. procedure DoVarOpInt64(var vl : TVarData; const vr : TVarData; const OpCode : TVarOp);
  1227. var
  1228. l, r : Int64;
  1229. Overflow : Boolean;
  1230. begin
  1231. l := VariantToInt64(vl);
  1232. r := VariantToInt64(vr);
  1233. Overflow := False;
  1234. case OpCode of
  1235. {$R+}{$Q+}
  1236. opAdd..opMultiply,opPower: try
  1237. case OpCode of
  1238. opAdd : l := l + r;
  1239. opSubtract : l := l - r;
  1240. opMultiply : l := l * r;
  1241. {$ifndef FPUNONE}
  1242. opPower : l := l ** r;
  1243. {$endif}
  1244. end;
  1245. except
  1246. on E: SysUtils.ERangeError do
  1247. Overflow := True;
  1248. on E: SysUtils.EIntOverflow do
  1249. Overflow := True;
  1250. end;
  1251. {$IFDEF RANGECHECKINGOFF} {$R-} {$ENDIF} {$IFDEF OVERFLOWCHECKINGOFF} {$Q+} {$ENDIF}
  1252. opIntDivide : l := l div r;
  1253. opModulus : l := l mod r;
  1254. opShiftLeft : l := l shl r;
  1255. opShiftRight : l := l shr r;
  1256. opAnd : l := l and r;
  1257. opOr : l := l or r;
  1258. opXor : l := l xor r;
  1259. else
  1260. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1261. end;
  1262. if Overflow then
  1263. DoVarOpFloat(vl,vr,OpCode)
  1264. else begin
  1265. DoVarClearIfComplex(vl);
  1266. vl.vType := varInt64;
  1267. vl.vInt64 := l;
  1268. end;
  1269. end;
  1270. procedure DoVarOpInt64to32(var vl : TVarData; const vr : TVarData; const OpCode : TVarOp);
  1271. begin
  1272. { can't do this well without an efficent way to check for overflows,
  1273. let the Int64 version handle it and check the Result if we can downgrade it
  1274. to integer }
  1275. DoVarOpInt64(vl, vr, OpCode);
  1276. with vl do
  1277. if (vType = varInt64) and (vInt64 >= Low(LongInt)) and (vInt64 <= High(LongInt)) then begin
  1278. vInteger := vInt64;
  1279. vType := varInteger;
  1280. end;
  1281. end;
  1282. procedure DoVarOpBool(var vl : TVarData; const vr : TVarData; const OpCode : TVarOp);
  1283. var
  1284. l,r: Boolean;
  1285. begin
  1286. l := VariantToBoolean(vl);
  1287. r := VariantToBoolean(vr);
  1288. case OpCode of
  1289. opAnd : l := l and r;
  1290. opOr : l := l or r;
  1291. opXor : l := l xor r;
  1292. else
  1293. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1294. end;
  1295. DoVarClearIfComplex(vl);
  1296. vl.vType := varBoolean;
  1297. vl.vBoolean := l;
  1298. end;
  1299. procedure DoVarOpNull(var vl : TVarData; const vr : TVarData; const OpCode : TVarOp);
  1300. begin
  1301. if (OpCode = opAnd) or (OpCode = opOr) then
  1302. if vl.vType = varNull then begin
  1303. if vr.vType = varNull then begin
  1304. {both null, do nothing }
  1305. end else begin
  1306. {Left null, Right not}
  1307. if OpCode = opAnd then begin
  1308. if not VariantToBoolean(vr) then
  1309. VarCopyProc(vl, vr);
  1310. end else {OpCode = opOr} begin
  1311. if VariantToBoolean(vr) then
  1312. VarCopyProc(vl, vr);
  1313. end;
  1314. end;
  1315. end else begin
  1316. if vr.vType = varNull then begin
  1317. {Right null, Left not}
  1318. if OpCode = opAnd then begin
  1319. if VariantToBoolean(vl) then begin
  1320. DoVarClearIfComplex(vl);
  1321. vl.vType := varNull;
  1322. end;
  1323. end else {OpCode = opOr} begin
  1324. if not VariantToBoolean(vl) then begin
  1325. DoVarClearIfComplex(vl);
  1326. vl.vType := varNull;
  1327. end;
  1328. end;
  1329. end else begin
  1330. { both not null, shouldn't happen }
  1331. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1332. end;
  1333. end
  1334. else begin
  1335. DoVarClearIfComplex(vl);
  1336. vl.vType := varNull;
  1337. end;
  1338. end;
  1339. procedure DoVarOpWStrCat(var vl : TVarData; const vr : TVarData);
  1340. var
  1341. ws: WideString;
  1342. begin
  1343. ws := VariantToWideString(vl) + VariantToWideString(vr);
  1344. DoVarClearIfComplex(vl);
  1345. vl.vType := varOleStr;
  1346. { transfer the WideString without making a copy }
  1347. Pointer(vl.vOleStr) := Pointer(ws);
  1348. { prevent the WideString from being freed, the reference has been transfered
  1349. from the local to the variant and will be correctly finalized when the
  1350. variant is finalized. }
  1351. Pointer(ws) := nil;
  1352. end;
  1353. procedure DoVarOpLStrCat(var vl: TVarData; const vr : TVarData);
  1354. var
  1355. s: AnsiString;
  1356. begin
  1357. s := VariantToAnsiString(vl) + VariantToAnsiString(vr);
  1358. DoVarClearIfComplex(vl);
  1359. vl.vType := varString;
  1360. { transfer the AnsiString without making a copy }
  1361. Pointer(vl.vString) := Pointer(s);
  1362. { prevent the AnsiString from being freed, the reference has been transfered
  1363. from the local to the variant and will be correctly finalized when the
  1364. variant is finalized. }
  1365. Pointer(s) := nil;
  1366. end;
  1367. procedure DoVarOpDate(var vl : TVarData; const vr : TVarData; const OpCode : TVarOp);
  1368. {$ifndef FPUNONE}
  1369. var
  1370. l, r : TDateTime;
  1371. begin
  1372. l := VariantToDate(vl);
  1373. r := VariantToDate(vr);
  1374. case OpCode of
  1375. opAdd : l := l + r;
  1376. opSubtract : l := l - r;
  1377. else
  1378. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1379. end;
  1380. DoVarClearIfComplex(vl);
  1381. vl.vType := varDate;
  1382. vl.vDate := l;
  1383. {$else}
  1384. begin
  1385. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1386. {$endif}
  1387. end;
  1388. procedure DoVarOpCurr(var vl : TVarData; const vr : TVarData; const OpCode : TVarOp; const lct, rct : TCommonType);
  1389. {$ifndef FPUNONE}
  1390. var
  1391. c : Currency;
  1392. d : Double;
  1393. begin
  1394. case OpCode of
  1395. opAdd:
  1396. c := VariantToCurrency(vl) + VariantToCurrency(vr);
  1397. opSubtract:
  1398. c := VariantToCurrency(vl) - VariantToCurrency(vr);
  1399. opMultiply:
  1400. if lct = ctCurrency then
  1401. if rct = ctCurrency then {both Currency}
  1402. c := VariantToCurrency(vl) * VariantToCurrency(vr)
  1403. else {Left Currency}
  1404. c := VariantToCurrency(vl) * VariantToDouble(vr)
  1405. else
  1406. if rct = ctCurrency then {rigth Currency}
  1407. c := VariantToDouble(vl) * VariantToCurrency(vr)
  1408. else {non Currency, error}
  1409. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1410. opDivide:
  1411. if lct = ctCurrency then
  1412. if rct = ctCurrency then {both Currency}
  1413. c := VariantToCurrency(vl) / VariantToCurrency(vr)
  1414. else {Left Currency}
  1415. c := VariantToCurrency(vl) / VariantToDouble(vr)
  1416. else
  1417. if rct = ctCurrency then begin {rigth Currency}
  1418. d := VariantToCurrency(vl) / VariantToCurrency(vr);
  1419. DoVarClearIfComplex(vl);
  1420. vl.vType := varDouble;
  1421. vl.vDouble := d;
  1422. Exit;
  1423. end else {non Currency, error}
  1424. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1425. opPower:
  1426. if lct = ctCurrency then
  1427. if rct = ctCurrency then {both Currency}
  1428. c := VariantToCurrency(vl) ** VariantToCurrency(vr)
  1429. else {Left Currency}
  1430. c := VariantToCurrency(vl) ** VariantToDouble(vr)
  1431. else
  1432. if rct = ctCurrency then {rigth Currency}
  1433. c := VariantToDouble(vl) ** VariantToCurrency(vr)
  1434. else {non Currency, error}
  1435. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1436. else
  1437. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1438. end;
  1439. DoVarClearIfComplex(vl);
  1440. vl.vType := varCurrency;
  1441. vl.vCurrency := c;
  1442. {$else}
  1443. begin
  1444. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1445. {$endif}
  1446. end;
  1447. procedure DoVarOpComplex(var vl : TVarData; const vr : TVarData; const OpCode : TVarOp);
  1448. var Handler: TCustomVariantType;
  1449. begin
  1450. if FindCustomVariantType(vl.vType, Handler) then
  1451. Handler.BinaryOp(vl, vr, OpCode)
  1452. else if FindCustomVariantType(vr.vType, Handler) then
  1453. Handler.BinaryOp(vl, vr, OpCode)
  1454. else
  1455. VarInvalidOp(vl.vType, vr.vType, OpCode);
  1456. end;
  1457. procedure SysVarOp(var Left : Variant; const Right : Variant; OpCode : TVarOp);
  1458. var
  1459. lct: TCommonType;
  1460. rct: TCommonType;
  1461. {$IFDEF DEBUG_VARIANTS}
  1462. i: Integer;
  1463. {$ENDIF}
  1464. begin
  1465. { as the function in cvarutil.inc can handle varByRef correctly we simply
  1466. resolve the final type }
  1467. lct := MapToCommonType(VarTypeDeRef(Left));
  1468. rct := MapToCommonType(VarTypeDeRef(Right));
  1469. {$IFDEF DEBUG_VARIANTS}
  1470. if __DEBUG_VARIANTS then begin
  1471. WriteLn('SysVarOp $', IntToHex(Cardinal(@TVarData(Left)),8), ' ', GetEnumName(TypeInfo(TVarOp), Ord(OpCode)) ,' $', IntToHex(Cardinal(@TVarData(Right)),8));
  1472. DumpVariant('SysVarOp/TVarData(Left)', TVarData(Left));
  1473. WriteLn('lct ', GetEnumName(TypeInfo(TCommonType), Ord(lct)));
  1474. DumpVariant('SysVarOp/TVarData(Right)', TVarData(Right));
  1475. WriteLn('rct ', GetEnumName(TypeInfo(TCommonType), Ord(rct)));
  1476. WriteLn('common ', GetEnumName(TypeInfo(TCommonType), Ord(FindOpCommonType[lct, rct])));
  1477. end;
  1478. {$ENDIF}
  1479. case FindOpCommonType[lct, rct] of
  1480. ctEmpty:
  1481. case OpCode of
  1482. opDivide:
  1483. Error(reZeroDivide);
  1484. opIntDivide, opModulus:
  1485. Error(reDivByZero);
  1486. else
  1487. DoVarClear(TVarData(Left));
  1488. end;
  1489. ctAny:
  1490. DoVarOpAny(TVarData(Left),TVarData(Right),OpCode);
  1491. ctLongInt:
  1492. case OpCode of
  1493. opAdd..opMultiply,opPower:
  1494. DoVarOpInt64to32(TVarData(Left),TVarData(Right),OpCode);
  1495. opDivide:
  1496. DoVarOpFloat(TVarData(Left),TVarData(Right),OpCode);
  1497. else
  1498. DoVarOpLongInt(TVarData(Left),TVarData(Right),OpCode);
  1499. end;
  1500. {$ifndef FPUNONE}
  1501. ctFloat:
  1502. if OpCode in [opAdd,opSubtract,opMultiply,opDivide] then
  1503. DoVarOpFloat(TVarData(Left),TVarData(Right),OpCode)
  1504. else
  1505. DoVarOpInt64to32(TVarData(Left),TVarData(Right),OpCode);
  1506. {$endif}
  1507. ctBoolean:
  1508. case OpCode of
  1509. opAdd..opMultiply, opPower:
  1510. DoVarOpFloat(TVarData(Left),TVarData(Right),OpCode);
  1511. opIntDivide..opShiftRight:
  1512. DoVarOpLongInt(TVarData(Left),TVarData(Right),OpCode);
  1513. opAnd..opXor:
  1514. DoVarOpBool(TVarData(Left),TVarData(Right),OpCode);
  1515. else
  1516. VarInvalidOp(TVarData(Left).vType, TVarData(Right).vType, OpCode);
  1517. end;
  1518. ctInt64:
  1519. if OpCode <> opDivide then
  1520. DoVarOpInt64(TVarData(Left),TVarData(Right),OpCode)
  1521. else
  1522. DoVarOpFloat(TVarData(Left),TVarData(Right),OpCode);
  1523. ctNull:
  1524. DoVarOpNull(TVarData(Left),TVarData(Right),OpCode);
  1525. ctWideStr:
  1526. case OpCode of
  1527. opAdd:
  1528. DoVarOpWStrCat(TVarData(Left),TVarData(Right));
  1529. opSubtract..opDivide,opPower:
  1530. DoVarOpFloat(TVarData(Left),TVarData(Right),OpCode);
  1531. opIntDivide..opXor:
  1532. DoVarOpInt64to32(TVarData(Left),TVarData(Right),OpCode);
  1533. else
  1534. VarInvalidOp(TVarData(Left).vType, TVarData(Right).vType, OpCode);
  1535. end;
  1536. {$ifndef FPUNONE}
  1537. ctDate:
  1538. case OpCode of
  1539. opAdd:
  1540. DoVarOpDate(TVarData(Left),TVarData(Right),OpCode);
  1541. opSubtract: begin
  1542. DoVarOpDate(TVarData(Left),TVarData(Right),OpCode);
  1543. if lct = rct then {both are date}
  1544. TVarData(Left).vType := varDouble;
  1545. end;
  1546. opMultiply, opDivide:
  1547. DoVarOpFloat(TVarData(Left),TVarData(Right),OpCode);
  1548. else
  1549. DoVarOpInt64to32(TVarData(Left),TVarData(Right),OpCode);
  1550. end;
  1551. ctCurrency:
  1552. if OpCode in [opAdd..opDivide, opPower] then
  1553. DoVarOpCurr(TVarData(Left),TVarData(Right),OpCode, lct, rct)
  1554. else
  1555. DoVarOpInt64to32(TVarData(Left),TVarData(Right),OpCode);
  1556. {$endif}
  1557. ctString:
  1558. case OpCode of
  1559. opAdd:
  1560. DoVarOpLStrCat(TVarData(Left),TVarData(Right));
  1561. opSubtract..opDivide,opPower:
  1562. DoVarOpFloat(TVarData(Left),TVarData(Right),OpCode);
  1563. opIntDivide..opXor:
  1564. DoVarOpInt64to32(TVarData(Left),TVarData(Right),OpCode);
  1565. else
  1566. VarInvalidOp(TVarData(Left).vType, TVarData(Right).vType, OpCode);
  1567. end;
  1568. else
  1569. { more complex case }
  1570. DoVarOpComplex(TVarData(Left),TVarData(Right),OpCode);
  1571. end;
  1572. end;
  1573. procedure DoVarNegAny(var v: TVarData);
  1574. begin
  1575. VarInvalidOp(v.vType, opNegate);
  1576. end;
  1577. procedure DoVarNegComplex(var v: TVarData);
  1578. begin
  1579. { custom variants? }
  1580. VarInvalidOp(v.vType, opNegate);
  1581. end;
  1582. procedure sysvarneg(var v: Variant);
  1583. const
  1584. BoolMap: array [Boolean] of SmallInt = (0, -1);
  1585. begin
  1586. with TVarData(v) do case vType of
  1587. varEmpty: begin
  1588. vSmallInt := 0;
  1589. vType := varSmallInt;
  1590. end;
  1591. varNull:;
  1592. varSmallint: vSmallInt := -vSmallInt;
  1593. varInteger: vInteger := -vInteger;
  1594. {$ifndef FPUNONE}
  1595. varSingle: vSingle := -vSingle;
  1596. varDouble: vDouble := -vDouble;
  1597. varCurrency: vCurrency := -vCurrency;
  1598. varDate: vDate := -vDate;
  1599. varOleStr: sysvarfromreal(v, -VariantToDouble(TVarData(v)));
  1600. {$else}
  1601. varOleStr: sysvarfromint64(v, -VariantToInt64(TVarData(v)));
  1602. {$endif}
  1603. varBoolean: begin
  1604. vSmallInt := BoolMap[vBoolean];
  1605. vType := varSmallInt;
  1606. end;
  1607. varShortInt: vShortInt := -vShortInt;
  1608. varByte: begin
  1609. vSmallInt := -vByte;
  1610. vType := varSmallInt;
  1611. end;
  1612. varWord: begin
  1613. vInteger := -vWord;
  1614. vType := varInteger;
  1615. end;
  1616. varLongWord:
  1617. if vLongWord and $80000000 <> 0 then begin
  1618. vInt64 := -vLongWord;
  1619. vType := varInt64;
  1620. end else begin
  1621. vInteger := -vLongWord;
  1622. vType := varInteger;
  1623. end;
  1624. varInt64: vInt64 := -vInt64;
  1625. varQWord: begin
  1626. if vQWord and $8000000000000000 <> 0 then
  1627. VarRangeCheckError(varQWord, varInt64);
  1628. vInt64 := -vQWord;
  1629. vType := varInt64;
  1630. end;
  1631. varVariant: v := -Variant(PVarData(vPointer)^);
  1632. else {with TVarData(v) do case vType of}
  1633. case vType of
  1634. {$ifndef FPUNONE}
  1635. varString: sysvarfromreal(v, -VariantToDouble(TVarData(v)));
  1636. {$else}
  1637. varString: sysvarfromint64(v, -VariantToInt64(TVarData(v)));
  1638. {$endif}
  1639. varAny: DoVarNegAny(TVarData(v));
  1640. else {case vType of}
  1641. if (vType and not varTypeMask) = varByRef then
  1642. case vType and varTypeMask of
  1643. varSmallInt: begin
  1644. vSmallInt := -PSmallInt(vPointer)^;
  1645. vType := varSmallInt;
  1646. end;
  1647. varInteger: begin
  1648. vInteger := -PInteger(vPointer)^;
  1649. vType := varInteger;
  1650. end;
  1651. {$ifndef FPUNONE}
  1652. varSingle: begin
  1653. vSingle := -PSingle(vPointer)^;
  1654. vType := varSingle;
  1655. end;
  1656. varDouble: begin
  1657. vDouble := -PDouble(vPointer)^;
  1658. vType := varDouble;
  1659. end;
  1660. varCurrency: begin
  1661. vCurrency := -PCurrency(vPointer)^;
  1662. vType := varCurrency;
  1663. end;
  1664. varDate: begin
  1665. vDate := -PDate(vPointer)^;
  1666. vType := varDate;
  1667. end;
  1668. varOleStr: sysvarfromreal(v, -VariantToDouble(TVarData(v)));
  1669. {$else}
  1670. varOleStr: sysvarfromint64(v, -VariantToInt64(TVarData(v)));
  1671. {$endif}
  1672. varBoolean: begin
  1673. vSmallInt := BoolMap[PWordBool(vPointer)^];
  1674. vType := varSmallInt;
  1675. end;
  1676. varShortInt: begin
  1677. vShortInt := -PShortInt(vPointer)^;
  1678. vType := varShortInt;
  1679. end;
  1680. varByte: begin
  1681. vSmallInt := -PByte(vPointer)^;
  1682. vType := varSmallInt;
  1683. end;
  1684. varWord: begin
  1685. vInteger := -PWord(vPointer)^;
  1686. vType := varInteger;
  1687. end;
  1688. varLongWord:
  1689. if PLongWord(vPointer)^ and $80000000 <> 0 then begin
  1690. vInt64 := -PLongWord(vPointer)^;
  1691. vType := varInt64;
  1692. end else begin
  1693. vInteger := -PLongWord(vPointer)^;
  1694. vType := varInteger;
  1695. end;
  1696. varInt64: begin
  1697. vInt64 := -PInt64(vPointer)^;
  1698. vType := varInt64;
  1699. end;
  1700. varQWord: begin
  1701. if PQWord(vPointer)^ and $8000000000000000 <> 0 then
  1702. VarRangeCheckError(varQWord, varInt64);
  1703. vInt64 := -PQWord(vPointer)^;
  1704. vType := varInt64;
  1705. end;
  1706. varVariant:
  1707. v := -Variant(PVarData(vPointer)^);
  1708. else {case vType and varTypeMask of}
  1709. DoVarNegComplex(TVarData(v));
  1710. end {case vType and varTypeMask of}
  1711. else {if (vType and not varTypeMask) = varByRef}
  1712. DoVarNegComplex(TVarData(v));
  1713. end; {case vType of}
  1714. end; {with TVarData(v) do case vType of}
  1715. end;
  1716. procedure DoVarNotAny(var v: TVarData);
  1717. begin
  1718. VarInvalidOp(v.vType, opNot);
  1719. end;
  1720. procedure DoVarNotOrdinal(var v: TVarData);
  1721. var
  1722. i: Int64;
  1723. begin
  1724. { only called for types that do no require finalization }
  1725. i := VariantToInt64(v);
  1726. with v do
  1727. if (i < Low(Integer)) or (i > High(Integer)) then begin
  1728. vInt64 := not i;
  1729. vType := varInt64;
  1730. end else begin
  1731. vInteger := not Integer(i);
  1732. vType := varInteger;
  1733. end
  1734. end;
  1735. procedure DoVarNotWStr(var v: TVarData; const p: Pointer);
  1736. var
  1737. i: Int64;
  1738. e: Word;
  1739. b: Boolean;
  1740. begin
  1741. Val(WideString(p), i, e);
  1742. with v do
  1743. if e = 0 then begin
  1744. DoVarClearIfComplex(v);
  1745. if (i < Low(Integer)) or (i > High(Integer)) then begin
  1746. vInt64 := not i;
  1747. vType := varInt64;
  1748. end else begin
  1749. vInteger := not Integer(i);
  1750. vType := varInteger;
  1751. end
  1752. end else begin
  1753. if not TryStrToBool(WideString(p), b) then
  1754. VarInvalidOp(vType, opNot);
  1755. DoVarClearIfComplex(v);
  1756. vBoolean := not b;
  1757. vType := varBoolean;
  1758. end;
  1759. end;
  1760. procedure DoVarNotLStr(var v: TVarData; const p: Pointer);
  1761. var
  1762. i: Int64;
  1763. e: Word;
  1764. b: Boolean;
  1765. begin
  1766. Val(AnsiString(p), i, e);
  1767. with v do
  1768. if e = 0 then begin
  1769. DoVarClearIfComplex(v);
  1770. if (i < Low(Integer)) or (i > High(Integer)) then begin
  1771. vInt64 := not i;
  1772. vType := varInt64;
  1773. end else begin
  1774. vInteger := not Integer(i);
  1775. vType := varInteger;
  1776. end
  1777. end else begin
  1778. if not TryStrToBool(AnsiString(p), b) then
  1779. VarInvalidOp(v.vType, opNot);
  1780. DoVarClearIfComplex(v);
  1781. vBoolean := not b;
  1782. vType := varBoolean;
  1783. end;
  1784. end;
  1785. procedure DoVarNotComplex(var v: TVarData);
  1786. begin
  1787. { custom variant support ?}
  1788. VarInvalidOp(v.vType, opNot);
  1789. end;
  1790. procedure sysvarnot(var v: Variant);
  1791. begin
  1792. with TVarData(v) do case vType of
  1793. varEmpty: v := -1;
  1794. varNull:;
  1795. varSmallint: vSmallInt := not vSmallInt;
  1796. varInteger: vInteger := not vInteger;
  1797. {$ifndef FPUNONE}
  1798. varSingle,
  1799. varDouble,
  1800. varCurrency,
  1801. varDate: DoVarNotOrdinal(TVarData(v));
  1802. {$endif}
  1803. varOleStr: DoVarNotWStr(TVarData(v), Pointer(vOleStr));
  1804. varBoolean: vBoolean := not vBoolean;
  1805. varShortInt: vShortInt := not vShortInt;
  1806. varByte: vByte := not vByte;
  1807. varWord: vWord := not vWord;
  1808. varLongWord: vLongWord := not vLongWord;
  1809. varInt64: vInt64 := not vInt64;
  1810. varQWord: vQWord := not vQWord;
  1811. varVariant: v := not Variant(PVarData(vPointer)^);
  1812. else {with TVarData(v) do case vType of}
  1813. case vType of
  1814. varString: DoVarNotLStr(TVarData(v), Pointer(vString));
  1815. varAny: DoVarNotAny(TVarData(v));
  1816. else {case vType of}
  1817. if (vType and not varTypeMask) = varByRef then
  1818. case vType and varTypeMask of
  1819. varSmallInt: begin
  1820. vSmallInt := not PSmallInt(vPointer)^;
  1821. vType := varSmallInt;
  1822. end;
  1823. varInteger: begin
  1824. vInteger := not PInteger(vPointer)^;
  1825. vType := varInteger;
  1826. end;
  1827. {$ifndef FPUNONE}
  1828. varSingle,
  1829. varDouble,
  1830. varCurrency,
  1831. varDate: DoVarNotOrdinal(TVarData(v));
  1832. {$endif}
  1833. varOleStr: DoVarNotWStr(TVarData(v), PPointer(vPointer)^);
  1834. varBoolean: begin
  1835. vBoolean := not PWordBool(vPointer)^;
  1836. vType := varBoolean;
  1837. end;
  1838. varShortInt: begin
  1839. vShortInt := not PShortInt(vPointer)^;
  1840. vType := varShortInt;
  1841. end;
  1842. varByte: begin
  1843. vByte := not PByte(vPointer)^;
  1844. vType := varByte;
  1845. end;
  1846. varWord: begin
  1847. vWord := not PWord(vPointer)^;
  1848. vType := varWord;
  1849. end;
  1850. varLongWord: begin
  1851. vLongWord := not PLongWord(vPointer)^;
  1852. vType := varLongWord;
  1853. end;
  1854. varInt64: begin
  1855. vInt64 := not PInt64(vPointer)^;
  1856. vType := varInt64;
  1857. end;
  1858. varQWord: begin
  1859. vQWord := not PQWord(vPointer)^;
  1860. vType := varQWord;
  1861. end;
  1862. varVariant:
  1863. v := not Variant(PVarData(vPointer)^);
  1864. else {case vType and varTypeMask of}
  1865. DoVarNotComplex(TVarData(v));
  1866. end {case vType and varTypeMask of}
  1867. else {if (vType and not varTypeMask) = varByRef}
  1868. DoVarNotComplex(TVarData(v));
  1869. end; {case vType of}
  1870. end; {with TVarData(v) do case vType of}
  1871. end;
  1872. {
  1873. This procedure is needed to destroy and clear non-standard variant type array elements,
  1874. which can not be handled by SafeArrayDestroy.
  1875. If array element type is varVariant, then clear each element individually before
  1876. calling VariantClear for array. VariantClear just calls SafeArrayDestroy.
  1877. }
  1878. procedure DoVarClearArray(var VArray: TVarData);
  1879. var
  1880. arr: pvararray;
  1881. i, cnt: cardinal;
  1882. data: pvardata;
  1883. begin
  1884. if VArray.vtype and varTypeMask = varVariant then begin
  1885. if WordBool(VArray.vType and varByRef) then
  1886. arr:=PVarArray(VArray.vPointer^)
  1887. else
  1888. arr:=VArray.vArray;
  1889. VarResultCheck(SafeArrayAccessData(arr, data));
  1890. try
  1891. { Calculation total number of elements in the array }
  1892. cnt:=1;
  1893. {$ifopt r+}
  1894. { arr^.bounds[] is an array[0..0] }
  1895. {$define rangeon}
  1896. {$r-}
  1897. {$endif}
  1898. for i:=0 to arr^.dimcount - 1 do
  1899. cnt:=cnt*cardinal(arr^.Bounds[i].ElementCount);
  1900. {$ifdef rangeon}
  1901. {$undef rangeon}
  1902. {$r+}
  1903. {$endif}
  1904. { Clearing each element }
  1905. for i:=1 to cnt do begin
  1906. DoVarClear(data^);
  1907. Inc(pointer(data), arr^.ElementSize);
  1908. end;
  1909. finally
  1910. VarResultCheck(SafeArrayUnaccessData(arr));
  1911. end;
  1912. end;
  1913. VariantClear(VArray);
  1914. end;
  1915. procedure DoVarClearComplex(var v : TVarData);
  1916. var
  1917. Handler : TCustomVariantType;
  1918. begin
  1919. with v do
  1920. if vType < varInt64 then
  1921. VarResultCheck(VariantClear(v))
  1922. else if vType = varString then begin
  1923. AnsiString(vString) := '';
  1924. vType := varEmpty
  1925. end else if vType = varAny then
  1926. ClearAnyProc(v)
  1927. else if vType and varArray <> 0 then
  1928. DoVarClearArray(v)
  1929. else if FindCustomVariantType(vType, Handler) then
  1930. Handler.Clear(v)
  1931. else begin
  1932. { ignore errors, if the OS doesn't know how to free it, we don't either }
  1933. VariantClear(v);
  1934. vType := varEmpty;
  1935. end;
  1936. end;
  1937. type
  1938. TVarArrayCopyCallback = procedure(var aDest: TVarData; const aSource: TVarData);
  1939. procedure DoVarCopyArray(var aDest: TVarData; const aSource: TVarData; aCallback: TVarArrayCopyCallback);
  1940. var
  1941. SourceArray : PVarArray;
  1942. SourcePtr : Pointer;
  1943. DestArray : PVarArray;
  1944. DestPtr : Pointer;
  1945. Bounds : array[0..63] of TVarArrayBound;
  1946. Iterator : TVariantArrayIterator;
  1947. Dims : Integer;
  1948. HighBound : Integer;
  1949. i : Integer;
  1950. begin
  1951. with aSource do begin
  1952. if vType and varArray = 0 then
  1953. VarResultCheck(VAR_INVALIDARG);
  1954. if (vType and varTypeMask) = varVariant then begin
  1955. if (vType and varByRef) <> 0 then
  1956. SourceArray := PVarArray(vPointer^)
  1957. else
  1958. SourceArray := vArray;
  1959. Dims := SourceArray^.DimCount;
  1960. for i := 0 to Pred(Dims) do
  1961. with Bounds[i] do begin
  1962. VarResultCheck(SafeArrayGetLBound(SourceArray, Succ(i), LowBound));
  1963. VarResultCheck(SafeArrayGetUBound(SourceArray, Succ(i), HighBound));
  1964. ElementCount := HighBound - LowBound + 1;
  1965. end;
  1966. DestArray := SafeArrayCreate(varVariant, Dims, PVarArrayBoundArray(@Bounds)^);
  1967. if not Assigned(DestArray) then
  1968. VarArrayCreateError;
  1969. DoVarClearIfComplex(aDest);
  1970. with aDest do begin
  1971. vType := varVariant or varArray;
  1972. vArray := DestArray;
  1973. end;
  1974. Iterator.Init(Dims, @Bounds);
  1975. try
  1976. if not(Iterator.AtEnd) then
  1977. repeat
  1978. VarResultCheck(SafeArrayPtrOfIndex(SourceArray, Iterator.Coords, SourcePtr));
  1979. VarResultCheck(SafeArrayPtrOfIndex(DestArray, Iterator.Coords, DestPtr));
  1980. aCallback(PVarData(DestPtr)^, PVarData(SourcePtr)^);
  1981. until not Iterator.Next;
  1982. finally
  1983. Iterator.Done;
  1984. end;
  1985. end else
  1986. VarResultCheck(VariantCopy(aDest, aSource));
  1987. end;
  1988. end;
  1989. procedure DoVarCopyComplex(var Dest: TVarData; const Source: TVarData);
  1990. var
  1991. Handler: TCustomVariantType;
  1992. begin
  1993. DoVarClearIfComplex(Dest);
  1994. with Source do
  1995. if vType < varInt64 then
  1996. VarResultCheck(VariantCopy(Dest, Source))
  1997. else if vType = varString then begin
  1998. Dest.vType := varString;
  1999. Dest.vString := nil;
  2000. AnsiString(Dest.vString) := AnsiString(vString);
  2001. end else if vType = varAny then begin
  2002. Dest := Source;
  2003. RefAnyProc(Dest);
  2004. end else if vType and varArray <> 0 then
  2005. DoVarCopyArray(Dest, Source, @DoVarCopy)
  2006. else if (vType and varByRef <> 0) and (vType xor varByRef = varString) then
  2007. Dest := Source
  2008. else if FindCustomVariantType(vType, Handler) then
  2009. Handler.Copy(Dest, Source, False)
  2010. else
  2011. VarResultCheck(VariantCopy(Dest, Source));
  2012. end;
  2013. procedure DoVarCopy(var Dest : TVarData; const Source : TVarData);
  2014. begin
  2015. if @Dest <> @Source then
  2016. if (Source.vType and varComplexType) = 0 then begin
  2017. DoVarClearIfComplex(Dest);
  2018. Dest := Source;
  2019. end else
  2020. DoVarCopyComplex(Dest, Source);
  2021. end;
  2022. procedure sysvarcopy (var Dest : Variant; const Source : Variant);
  2023. begin
  2024. DoVarCopy(TVarData(Dest),TVarData(Source));
  2025. end;
  2026. procedure DoVarAddRef(var v : TVarData); inline;
  2027. var
  2028. Dummy : TVarData;
  2029. begin
  2030. Dummy := v;
  2031. v.vType := varEmpty;
  2032. DoVarCopy(v, Dummy);
  2033. end;
  2034. procedure sysvaraddref(var v : Variant);
  2035. begin
  2036. DoVarAddRef(TVarData(v));
  2037. end;
  2038. procedure DoVarCastWStr(var aDest : TVarData; const aSource : TVarData);
  2039. begin
  2040. SysVarFromWStr(Variant(aDest), VariantToWideString(aSource));
  2041. end;
  2042. procedure DoVarCastLStr(var aDest : TVarData; const aSource : TVarData);
  2043. begin
  2044. SysVarFromLStr(Variant(aDest), VariantToAnsiString(aSource));
  2045. end;
  2046. procedure DoVarCastDispatch(var aDest : TVarData; const aSource : TVarData);
  2047. var
  2048. Disp: IDispatch;
  2049. begin
  2050. SysVarToDisp(Disp, Variant(aSource));
  2051. SysVarFromDisp(Variant(aDest), Disp);
  2052. end;
  2053. procedure DoVarCastInterface(var aDest : TVarData; const aSource : TVarData);
  2054. var
  2055. Intf: IInterface;
  2056. begin
  2057. SysVarToIntf(Intf, Variant(aSource));
  2058. SysVarFromIntf(Variant(aDest), Intf);
  2059. end;
  2060. procedure DoVarCastAny(var aDest : TVarData; const aSource : TVarData; aVarType : LongInt);
  2061. begin
  2062. VarCastError(aSource.vType, aVarType)
  2063. end;
  2064. procedure DoVarCastFallback(var aDest : TVarData; const aSource : TVarData; aVarType : LongInt);
  2065. begin
  2066. if aSource.vType and varTypeMask >= varInt64 then begin
  2067. DoVarCast(aDest, aSource, varOleStr);
  2068. VarResultCheck(VariantChangeTypeEx(aDest, aDest, VAR_LOCALE_USER_DEFAULT,
  2069. 0, aVarType), aSource.vType, aVarType);
  2070. end else if aVarType and varTypeMask < varInt64 then
  2071. VarResultCheck(VariantChangeTypeEx(aDest, aSource, VAR_LOCALE_USER_DEFAULT,
  2072. 0, aVarType), aSource.vType, aVarType)
  2073. else
  2074. VarCastError(aSource.vType, aVarType);
  2075. end;
  2076. procedure DoVarCastComplex(var aDest : TVarData; const aSource : TVarData; aVarType : LongInt);
  2077. var
  2078. Handler: TCustomVariantType;
  2079. begin
  2080. if aSource.vType = varAny then
  2081. DoVarCastAny(aDest, aSource, aVarType)
  2082. else if FindCustomVariantType(aSource.vType, Handler) then
  2083. Handler.CastTo(aDest, aSource, aVarType)
  2084. else if FindCustomVariantType(aVarType, Handler) then
  2085. Handler.Cast(aDest, aSource)
  2086. else
  2087. DoVarCastFallback(aDest, aSource, aVarType);
  2088. end;
  2089. procedure DoVarCast(var aDest : TVarData; const aSource : TVarData; aVarType : LongInt);
  2090. begin
  2091. with aSource do
  2092. if vType = aVarType then
  2093. DoVarCopy(aDest, aSource)
  2094. else begin
  2095. if (vType = varNull) and NullStrictConvert then
  2096. VarCastError(varNull, aVarType);
  2097. case aVarType of
  2098. varEmpty, varNull: begin
  2099. DoVarClearIfComplex(aDest);
  2100. aDest.vType := aVarType;
  2101. end;
  2102. varSmallInt: SysVarFromInt(Variant(aDest), VariantToSmallInt(aSource), -2);
  2103. varInteger: SysVarFromInt(Variant(aDest), VariantToLongInt(aSource), -4);
  2104. {$ifndef FPUNONE}
  2105. varSingle: SysVarFromSingle(Variant(aDest), VariantToSingle(aSource));
  2106. varDouble: SysVarFromDouble(Variant(aDest), VariantToDouble(aSource));
  2107. varCurrency: SysVarFromCurr(Variant(aDest), VariantToCurrency(aSource));
  2108. varDate: SysVarFromTDateTime(Variant(aDest), VariantToDate(aSource));
  2109. {$endif}
  2110. varOleStr: DoVarCastWStr(aDest, aSource);
  2111. varBoolean: SysVarFromBool(Variant(aDest), VariantToBoolean(aSource));
  2112. varShortInt: SysVarFromInt(Variant(aDest), VariantToShortInt(aSource), -1);
  2113. varByte: SysVarFromInt(Variant(aDest), VariantToByte(aSource), 1);
  2114. varWord: SysVarFromInt(Variant(aDest), VariantToLongInt(aSource), 2);
  2115. varLongWord: SysVarFromInt(Variant(aDest), Integer(VariantToCardinal(aSource)), 4);
  2116. varInt64: SysVarFromInt64(Variant(aDest), VariantToInt64(aSource));
  2117. varQWord: SysVarFromWord64(Variant(aDest), VariantToQWord(aSource));
  2118. varDispatch: DoVarCastDispatch(aDest, aSource);
  2119. varUnknown: DoVarCastInterface(aDest, aSource);
  2120. else
  2121. case aVarType of
  2122. varString: DoVarCastLStr(aDest, aSource);
  2123. varAny: VarCastError(vType, varAny);
  2124. else
  2125. DoVarCastComplex(aDest, aSource, aVarType);
  2126. end;
  2127. end;
  2128. end;
  2129. end;
  2130. procedure sysvarcast (var aDest : Variant; const aSource : Variant; aVarType : LongInt);
  2131. begin
  2132. DoVarCast(TVarData(aDest), TVarData(aSource), aVarType);
  2133. end;
  2134. procedure sysvarfromdynarray(var Dest : Variant; const Source : Pointer; TypeInfo: Pointer);
  2135. begin
  2136. DynArrayToVariant(Dest,Source,TypeInfo);
  2137. if VarIsEmpty(Dest) then
  2138. VarCastError;
  2139. end;
  2140. procedure sysolevarfrompstr(var Dest : olevariant; const Source : ShortString);
  2141. begin
  2142. sysvarfromwstr(Variant(TVarData(Dest)), Source);
  2143. end;
  2144. procedure sysolevarfromlstr(var Dest : olevariant; const Source : AnsiString);
  2145. begin
  2146. sysvarfromwstr(Variant(TVarData(Dest)), Source);
  2147. end;
  2148. procedure DoOleVarFromAny(var aDest : TVarData; const aSource : TVarData);
  2149. begin
  2150. VarCastErrorOle(aSource.vType);
  2151. end;
  2152. procedure DoOleVarFromVar(var aDest : TVarData; const aSource : TVarData);
  2153. var
  2154. Handler: TCustomVariantType;
  2155. begin
  2156. with aSource do
  2157. if vType = varByRef or varVariant then
  2158. DoOleVarFromVar(aDest, PVarData(vPointer)^)
  2159. else begin
  2160. case vType of
  2161. varShortInt, varByte, varWord:
  2162. DoVarCast(aDest, aSource, varInteger);
  2163. varLongWord:
  2164. if vLongWord and $80000000 = 0 then
  2165. DoVarCast(aDest, aSource, varInteger)
  2166. else
  2167. {$ifndef FPUNONE}
  2168. if OleVariantInt64AsDouble then
  2169. DoVarCast(aDest, aSource, varDouble)
  2170. else
  2171. {$endif}
  2172. DoVarCast(aDest, aSource, varInt64);
  2173. varInt64:
  2174. if (vInt64 < Low(Integer)) or (vInt64 > High(Integer)) then
  2175. {$ifndef FPUNONE}
  2176. if OleVariantInt64AsDouble then
  2177. DoVarCast(aDest, aSource, varDouble)
  2178. else
  2179. {$endif}
  2180. DoVarCast(aDest, aSource, varInt64)
  2181. else
  2182. DoVarCast(aDest, aSource, varInteger);
  2183. varQWord:
  2184. if vQWord > High(Integer) then
  2185. {$ifndef FPUNONE}
  2186. if OleVariantInt64AsDouble or (vQWord and $8000000000000000 <> 0) then
  2187. DoVarCast(aDest, aSource, varDouble)
  2188. else
  2189. {$endif}
  2190. DoVarCast(aDest, aSource, varInt64)
  2191. else
  2192. DoVarCast(aDest, aSource, varInteger);
  2193. varString:
  2194. DoVarCast(aDest, aSource, varOleStr);
  2195. varAny:
  2196. DoOleVarFromAny(aDest, aSource);
  2197. else
  2198. if (vType and varArray) <> 0 then
  2199. DoVarCopyArray(aDest, aSource, @DoOleVarFromVar)
  2200. else if (vType and varTypeMask) < CFirstUserType then
  2201. DoVarCopy(aDest, aSource)
  2202. else if FindCustomVariantType(vType, Handler) then
  2203. Handler.CastToOle(aDest, aSource)
  2204. else
  2205. VarCastErrorOle(vType);
  2206. end;
  2207. end;
  2208. end;
  2209. procedure sysolevarfromvar(var aDest : OleVariant; const aSource : Variant);
  2210. begin
  2211. DoOleVarFromVar(TVarData(aDest), TVarData(aSource));
  2212. end;
  2213. procedure sysolevarfromint(var Dest : olevariant; const Source : LongInt; const range : ShortInt);
  2214. begin
  2215. DoVarClearIfComplex(TVarData(Dest));
  2216. with TVarData(Dest) do begin
  2217. vInteger := Source;
  2218. vType := varInteger;
  2219. end;
  2220. end;
  2221. procedure DoVarCastOle(var aDest: TVarData; const aSource: TVarData; aVarType: LongInt);
  2222. var
  2223. Handler: TCustomVariantType;
  2224. begin
  2225. with aSource do
  2226. if vType = varByRef or varVariant then
  2227. DoVarCastOle(aDest, PVarData(VPointer)^, aVarType)
  2228. else
  2229. if (aVarType = varString) or (aVarType = varAny) then
  2230. VarCastError(vType, aVarType)
  2231. else if FindCustomVariantType(vType, Handler) then
  2232. Handler.CastTo(aDest, aSource, aVarType)
  2233. else
  2234. DoVarCast(aDest, aSource, aVarType);
  2235. end;
  2236. procedure sysvarcastole(var Dest : Variant; const Source : Variant; aVarType : LongInt);
  2237. begin
  2238. DoVarCastOle(TVarData(Dest), TVarData(Source), aVarType);
  2239. end;
  2240. procedure sysdispinvoke(Dest : PVarData; const Source : TVarData;calldesc : pcalldesc;params : Pointer);cdecl;
  2241. var
  2242. temp : TVarData;
  2243. tempp : ^TVarData;
  2244. customvarianttype : TCustomVariantType;
  2245. begin
  2246. if Source.vType=(varByRef or varVariant) then
  2247. sysdispinvoke(Dest,PVarData(Source.vPointer)^,calldesc,params)
  2248. else
  2249. begin
  2250. try
  2251. { get a defined Result }
  2252. if not(assigned(Dest)) then
  2253. tempp:=nil
  2254. else
  2255. begin
  2256. fillchar(temp,SizeOf(temp),0);
  2257. tempp:=@temp;
  2258. end;
  2259. case Source.vType of
  2260. varDispatch,
  2261. varAny,
  2262. varUnknown,
  2263. varDispatch or varByRef,
  2264. varAny or varByRef,
  2265. varUnknown or varByRef:
  2266. VarDispProc(pvariant(tempp),Variant(Source),calldesc,params);
  2267. else
  2268. begin
  2269. if FindCustomVariantType(Source.vType,customvarianttype) then
  2270. customvarianttype.DispInvoke(tempp,Source,calldesc,params)
  2271. else
  2272. VarInvalidOp;
  2273. end;
  2274. end;
  2275. finally
  2276. if assigned(tempp) then
  2277. begin
  2278. DoVarCopy(Dest^,tempp^);
  2279. DoVarClear(temp);
  2280. end;
  2281. end;
  2282. end;
  2283. end;
  2284. procedure sysvararrayredim(var a : Variant;highbound : SizeInt);
  2285. var
  2286. src : TVarData;
  2287. p : pvararray;
  2288. newbounds : tvararraybound;
  2289. begin
  2290. src:=TVarData(a);
  2291. { get final Variant }
  2292. while src.vType=varByRef or varVariant do
  2293. src:=TVarData(src.vPointer^);
  2294. if (src.vType and varArray)<>0 then
  2295. begin
  2296. { get Pointer to the array }
  2297. if (src.vType and varByRef)<>0 then
  2298. p:=pvararray(src.vPointer^)
  2299. else
  2300. p:=src.vArray;
  2301. {$ifopt r+}
  2302. {$define rangeon}
  2303. {$r-}
  2304. {$endif}
  2305. if highbound<p^.Bounds[p^.dimcount-1].LowBound-1 then
  2306. VarInvalidArgError;
  2307. newbounds.LowBound:=p^.Bounds[p^.dimcount-1].LowBound;
  2308. {$ifdef rangon}
  2309. {$undef rangeon}
  2310. {$r+}
  2311. {$endif}
  2312. newbounds.ElementCount:=highbound-newbounds.LowBound+1;
  2313. VarResultCheck(SafeArrayRedim(p,newbounds));
  2314. end
  2315. else
  2316. VarInvalidArgError(src.vType);
  2317. end;
  2318. function getfinalvartype(const v : TVarData) : TVarType;{$IFDEF VARIANTINLINE}inline;{$ENDIF VARIANTINLINE}
  2319. var
  2320. p: PVarData;
  2321. begin
  2322. p := @v;
  2323. while p^.vType = varByRef or varVariant do
  2324. p := PVarData(p^.vPointer);
  2325. Result := p^.vType;
  2326. end;
  2327. function sysvararrayget(const a : Variant;indexcount : SizeInt;indices : plongint) : Variant;cdecl;
  2328. var
  2329. src : TVarData;
  2330. p : pvararray;
  2331. arraysrc : pvariant;
  2332. arrayelementtype : TVarType;
  2333. begin
  2334. src:=TVarData(a);
  2335. { get final Variant }
  2336. while src.vType=varByRef or varVariant do
  2337. src:=TVarData(src.vPointer^);
  2338. if (src.vType and varArray)<>0 then
  2339. begin
  2340. { get Pointer to the array }
  2341. if (src.vType and varByRef)<>0 then
  2342. p:=pvararray(src.vPointer^)
  2343. else
  2344. p:=src.vArray;
  2345. { number of indices ok? }
  2346. if p^.DimCount<>indexcount then
  2347. VarInvalidArgError;
  2348. arrayelementtype:=src.vType and varTypeMask;
  2349. if arrayelementtype=varVariant then
  2350. begin
  2351. VarResultCheck(SafeArrayPtrOfIndex(p,PVarArrayCoorArray(indices),arraysrc));
  2352. Result:=arraysrc^;
  2353. end
  2354. else
  2355. begin
  2356. TVarData(Result).vType:=arrayelementtype;
  2357. VarResultCheck(SafeArrayGetElement(p,PVarArrayCoorArray(indices),@TVarData(Result).vPointer));
  2358. end;
  2359. end
  2360. else
  2361. VarInvalidArgError(src.vType);
  2362. end;
  2363. procedure sysvararrayput(var a : Variant; const value : Variant;indexcount : SizeInt;indices : plongint);cdecl;
  2364. var
  2365. Dest : TVarData;
  2366. p : pvararray;
  2367. arraydest : pvariant;
  2368. valuevtype,
  2369. arrayelementtype : TVarType;
  2370. tempvar : Variant;
  2371. variantmanager : tvariantmanager;
  2372. begin
  2373. Dest:=TVarData(a);
  2374. { get final Variant }
  2375. while Dest.vType=varByRef or varVariant do
  2376. Dest:=TVarData(Dest.vPointer^);
  2377. valuevtype:=getfinalvartype(TVarData(value));
  2378. if not(VarTypeIsValidElementType(valuevtype)) and
  2379. { varString isn't a valid varArray type but it is converted
  2380. later }
  2381. (valuevtype<>varString) then
  2382. VarCastError(valuevtype,Dest.vType);
  2383. if (Dest.vType and varArray)<>0 then
  2384. begin
  2385. { get Pointer to the array }
  2386. if (Dest.vType and varByRef)<>0 then
  2387. p:=pvararray(Dest.vPointer^)
  2388. else
  2389. p:=Dest.vArray;
  2390. { number of indices ok? }
  2391. if p^.DimCount<>indexcount then
  2392. VarInvalidArgError;
  2393. arrayelementtype:=Dest.vType and varTypeMask;
  2394. if arrayelementtype=varVariant then
  2395. begin
  2396. VarResultCheck(SafeArrayPtrOfIndex(p,PVarArrayCoorArray(indices),arraydest));
  2397. { we can't store ansistrings in Variant arrays so we convert the string to
  2398. an olestring }
  2399. if valuevtype=varString then
  2400. begin
  2401. tempvar:=VarToWideStr(value);
  2402. arraydest^:=tempvar;
  2403. end
  2404. else
  2405. arraydest^:=value;
  2406. end
  2407. else
  2408. begin
  2409. GetVariantManager(variantmanager);
  2410. variantmanager.varcast(tempvar,value,arrayelementtype);
  2411. if arrayelementtype in [varOleStr,varDispatch,varUnknown] then
  2412. VarResultCheck(SafeArrayPutElement(p,PVarArrayCoorArray(indices),TVarData(tempvar).vPointer))
  2413. else
  2414. VarResultCheck(SafeArrayPutElement(p,PVarArrayCoorArray(indices),@TVarData(tempvar).vPointer));
  2415. end;
  2416. end
  2417. else
  2418. VarInvalidArgError(Dest.vType);
  2419. end;
  2420. { import from system unit }
  2421. Procedure fpc_Write_Text_AnsiStr (Len : LongInt; Var f : Text; S : AnsiString); external name 'FPC_WRITE_TEXT_ANSISTR';
  2422. function syswritevariant(var t : text; const v : Variant;width : LongInt) : Pointer;
  2423. var
  2424. s : AnsiString;
  2425. variantmanager : tvariantmanager;
  2426. begin
  2427. GetVariantManager(variantmanager);
  2428. variantmanager.vartolstr(s,v);
  2429. fpc_write_text_ansistr(width,t,s);
  2430. Result:=nil; // Pointer to what should be returned?
  2431. end;
  2432. function syswrite0Variant(var t : text; const v : Variant) : Pointer;
  2433. var
  2434. s : AnsiString;
  2435. variantmanager : tvariantmanager;
  2436. begin
  2437. getVariantManager(variantmanager);
  2438. variantmanager.vartolstr(s,v);
  2439. fpc_write_text_ansistr(-1,t,s);
  2440. Result:=nil; // Pointer to what should be returned?
  2441. end;
  2442. Const
  2443. SysVariantManager : TVariantManager = (
  2444. vartoint : @sysvartoint;
  2445. vartoint64 : @sysvartoint64;
  2446. vartoword64 : @sysvartoword64;
  2447. vartobool : @sysvartobool;
  2448. {$ifndef FPUNONE}
  2449. vartoreal : @sysvartoreal;
  2450. vartotdatetime: @sysvartotdatetime;
  2451. {$endif}
  2452. vartocurr : @sysvartocurr;
  2453. vartopstr : @sysvartopstr;
  2454. vartolstr : @sysvartolstr;
  2455. vartowstr : @sysvartowstr;
  2456. vartointf : @sysvartointf;
  2457. vartodisp : @sysvartodisp;
  2458. vartodynarray : @sysvartodynarray;
  2459. varfrombool : @sysvarfromBool;
  2460. varfromint : @sysvarfromint;
  2461. varfromint64 : @sysvarfromint64;
  2462. varfromword64 : @sysvarfromword64;
  2463. {$ifndef FPUNONE}
  2464. varfromreal : @sysvarfromreal;
  2465. varfromtdatetime: @sysvarfromtdatetime;
  2466. {$endif}
  2467. varfromcurr : @sysvarfromcurr;
  2468. varfrompstr : @sysvarfrompstr;
  2469. varfromlstr : @sysvarfromlstr;
  2470. varfromwstr : @sysvarfromwstr;
  2471. varfromintf : @sysvarfromintf;
  2472. varfromdisp : @sysvarfromdisp;
  2473. varfromdynarray: @sysvarfromdynarray;
  2474. olevarfrompstr: @sysolevarfrompstr;
  2475. olevarfromlstr: @sysolevarfromlstr;
  2476. olevarfromvar : @sysolevarfromvar;
  2477. olevarfromint : @sysolevarfromint;
  2478. varop : @SysVarOp;
  2479. cmpop : @syscmpop;
  2480. varneg : @sysvarneg;
  2481. varnot : @sysvarnot;
  2482. varinit : @sysvarinit;
  2483. varclear : @sysvarclear;
  2484. varaddref : @sysvaraddref;
  2485. varcopy : @sysvarcopy;
  2486. varcast : @sysvarcast;
  2487. varcastole : @sysvarcastole;
  2488. dispinvoke : @sysdispinvoke;
  2489. vararrayredim : @sysvararrayredim;
  2490. vararrayget : @sysvararrayget;
  2491. vararrayput : @sysvararrayput;
  2492. writevariant : @syswritevariant;
  2493. write0Variant : @syswrite0variant;
  2494. );
  2495. Var
  2496. PrevVariantManager : TVariantManager;
  2497. Procedure SetSysVariantManager;
  2498. begin
  2499. GetVariantManager(PrevVariantManager);
  2500. SetVariantManager(SysVariantManager);
  2501. end;
  2502. Procedure UnsetSysVariantManager;
  2503. begin
  2504. SetVariantManager(PrevVariantManager);
  2505. end;
  2506. { ---------------------------------------------------------------------
  2507. Variant support procedures and functions
  2508. ---------------------------------------------------------------------}
  2509. function VarType(const V: Variant): TVarType;
  2510. begin
  2511. Result:=TVarData(V).vType;
  2512. end;
  2513. function VarTypeDeRef(const V: Variant): TVarType;
  2514. var
  2515. p: PVarData;
  2516. begin
  2517. p := @TVarData(V);
  2518. Result := p^.vType and not varByRef;
  2519. while Result = varVariant do begin
  2520. p := p^.vPointer;
  2521. if not Assigned(p) then
  2522. VarBadTypeError;
  2523. Result := p^.vType and not varByRef;
  2524. end;
  2525. end;
  2526. function VarTypeDeRef(const V: TVarData): TVarType;
  2527. begin
  2528. Result := VarTypeDeRef(Variant(v));
  2529. end;
  2530. function VarAsType(const V: Variant; aVarType: TVarType): Variant;
  2531. begin
  2532. sysvarcast(Result,V,aVarType);
  2533. end;
  2534. function VarIsType(const V: Variant; aVarType: TVarType): Boolean; overload;
  2535. begin
  2536. Result:=((TVarData(V).vType and varTypeMask)=aVarType);
  2537. end;
  2538. function VarIsType(const V: Variant; const AVarTypes: array of TVarType): Boolean; overload;
  2539. Var
  2540. I : Integer;
  2541. begin
  2542. I:=Low(AVarTypes);
  2543. Result:=False;
  2544. While Not Result and (I<=High(AVarTypes)) do
  2545. begin
  2546. Result:=((TVarData(V).vType and varTypeMask)=AVarTypes[I]);
  2547. inc(i);
  2548. end;
  2549. end;
  2550. function VarIsByRef(const V: Variant): Boolean;
  2551. begin
  2552. Result:=(TVarData(V).vType and varByRef)<>0;
  2553. end;
  2554. function VarIsEmpty(const V: Variant): Boolean;
  2555. begin
  2556. Result:=TVarData(V).vType=varEmpty;
  2557. end;
  2558. procedure VarCheckEmpty(const V: Variant);
  2559. begin
  2560. If VarIsEmpty(V) Then
  2561. VariantError(SErrVarIsEmpty);
  2562. end;
  2563. procedure VarClear(var V: Variant);{$IFDEF VARIANTINLINE}inline;{$ENDIF VARIANTINLINE}
  2564. begin
  2565. sysvarclear(v);
  2566. end;
  2567. procedure VarClear(var V: OleVariant);{$IFDEF VARIANTINLINE}inline;{$ENDIF VARIANTINLINE}
  2568. begin
  2569. { strange casting using TVarData to avoid call of helper olevariant->Variant }
  2570. sysvarclear(Variant(TVarData(v)));
  2571. end;
  2572. function VarIsNull(const V: Variant): Boolean;
  2573. begin
  2574. Result:=TVarData(V).vType=varNull;
  2575. end;
  2576. function VarIsClear(const V: Variant): Boolean;
  2577. Var
  2578. VT : TVarType;
  2579. begin
  2580. VT:=TVarData(V).vType and varTypeMask;
  2581. Result:=(VT=varEmpty) or
  2582. (((VT=varDispatch) or (VT=varUnknown))
  2583. and (TVarData(V).vDispatch=Nil));
  2584. end;
  2585. function VarIsCustom(const V: Variant): Boolean;
  2586. begin
  2587. Result:=TVarData(V).vType>=CFirstUserType;
  2588. end;
  2589. function VarIsOrdinal(const V: Variant): Boolean;
  2590. begin
  2591. Result:=(TVarData(V).vType and varTypeMask) in OrdinalVarTypes;
  2592. end;
  2593. function VarIsFloat(const V: Variant): Boolean;
  2594. begin
  2595. Result:=(TVarData(V).vType and varTypeMask) in FloatVarTypes;
  2596. end;
  2597. function VarIsNumeric(const V: Variant): Boolean;
  2598. begin
  2599. Result:=(TVarData(V).vType and varTypeMask) in (OrdinalVarTypes + FloatVarTypes);
  2600. end;
  2601. function VarIsStr(const V: Variant): Boolean;
  2602. begin
  2603. case (TVarData(V).vType and varTypeMask) of
  2604. varOleStr,
  2605. varString :
  2606. Result:=True;
  2607. else
  2608. Result:=False;
  2609. end;
  2610. end;
  2611. function VarToStr(const V: Variant): string;
  2612. begin
  2613. Result:=VarToStrDef(V,'');
  2614. end;
  2615. function VarToStrDef(const V: Variant; const ADefault: string): string;
  2616. begin
  2617. If TVarData(V).vType<>varNull then
  2618. Result:=V
  2619. else
  2620. Result:=ADefault;
  2621. end;
  2622. function VarToWideStr(const V: Variant): WideString;
  2623. begin
  2624. Result:=VarToWideStrDef(V,'');
  2625. end;
  2626. function VarToWideStrDef(const V: Variant; const ADefault: WideString): WideString;
  2627. begin
  2628. If TVarData(V).vType<>varNull then
  2629. Result:=V
  2630. else
  2631. Result:=ADefault;
  2632. end;
  2633. function VarToUnicodeStr(const V: Variant): UnicodeString;
  2634. begin
  2635. Result:=VarToUnicodeStrDef(V,'');
  2636. end;
  2637. function VarToUnicodeStrDef(const V: Variant; const ADefault: UnicodeString): UnicodeString;
  2638. begin
  2639. If TVarData(V).vType<>varNull then
  2640. Result:=V
  2641. else
  2642. Result:=ADefault;
  2643. end;
  2644. {$ifndef FPUNONE}
  2645. function VarToDateTime(const V: Variant): TDateTime;
  2646. begin
  2647. Result:=VariantToDate(TVarData(V));
  2648. end;
  2649. function VarFromDateTime(const DateTime: TDateTime): Variant;
  2650. begin
  2651. SysVarClear(Result);
  2652. with TVarData(Result) do
  2653. begin
  2654. vType:=varDate;
  2655. vdate:=DateTime;
  2656. end;
  2657. end;
  2658. {$endif}
  2659. function VarInRange(const AValue, AMin, AMax: Variant): Boolean;
  2660. begin
  2661. Result:=(AValue>=AMin) and (AValue<=AMax);
  2662. end;
  2663. function VarEnsureRange(const AValue, AMin, AMax: Variant): Variant;
  2664. begin
  2665. If Result>AMAx then
  2666. Result:=AMax
  2667. else If Result<AMin Then
  2668. Result:=AMin
  2669. else
  2670. Result:=AValue;
  2671. end;
  2672. function VarSameValue(const A, B: Variant): Boolean;
  2673. var
  2674. v1,v2 : TVarData;
  2675. begin
  2676. v1:=FindVarData(a)^;
  2677. v2:=FindVarData(b)^;
  2678. if v1.vType in [varEmpty,varNull] then
  2679. Result:=v1.vType=v2.vType
  2680. else if v2.vType in [varEmpty,varNull] then
  2681. Result:=False
  2682. else
  2683. Result:=A=B;
  2684. end;
  2685. function VarCompareValue(const A, B: Variant): TVariantRelationship;
  2686. var
  2687. v1,v2 : TVarData;
  2688. begin
  2689. Result:=vrNotEqual;
  2690. v1:=FindVarData(a)^;
  2691. v2:=FindVarData(b)^;
  2692. if (v1.vType in [varEmpty,varNull]) and (v1.vType=v2.vType) then
  2693. Result:=vrEqual
  2694. else if not(v2.vType in [varEmpty,varNull]) and
  2695. not(v1.vType in [varEmpty,varNull]) then
  2696. begin
  2697. if a=b then
  2698. Result:=vrEqual
  2699. else if a>b then
  2700. Result:=vrGreaterThan
  2701. else
  2702. Result:=vrLessThan;
  2703. end;
  2704. end;
  2705. function VarIsEmptyParam(const V: Variant): Boolean;
  2706. begin
  2707. Result:=(TVarData(V).vType = varError) and
  2708. (TVarData(V).vError=VAR_PARAMNOTFOUND);
  2709. end;
  2710. procedure SetClearVarToEmptyParam(var V: TVarData);
  2711. begin
  2712. VariantClear(V);
  2713. V.vType := varError;
  2714. V.vError := VAR_PARAMNOTFOUND;
  2715. end;
  2716. function VarIsError(const V: Variant; out aResult: HRESULT): Boolean;
  2717. begin
  2718. Result := TVarData(V).vType = varError;
  2719. if Result then
  2720. aResult := TVarData(v).vError;
  2721. end;
  2722. function VarIsError(const V: Variant): Boolean;
  2723. begin
  2724. Result := TVarData(V).vType = varError;
  2725. end;
  2726. function VarAsError(AResult: HRESULT): Variant;
  2727. begin
  2728. TVarData(Result).vType:=varError;
  2729. TVarData(Result).vError:=AResult;
  2730. end;
  2731. function VarSupports(const V: Variant; const IID: TGUID; out Intf): Boolean;
  2732. begin
  2733. case TVarData(v).vType of
  2734. varUnknown:
  2735. Result := Assigned(TVarData(v).vUnknown) and (IInterface(TVarData(v).vUnknown).QueryInterface(IID, Intf) = S_OK);
  2736. varUnknown or varByRef:
  2737. Result := Assigned(TVarData(v).vPointer) and Assigned(pointer(TVarData(v).vPointer^)) and (IInterface(TVarData(v).vPointer^).QueryInterface(IID, Intf) = S_OK);
  2738. varDispatch:
  2739. Result := Assigned(TVarData(v).vDispatch) and (IInterface(TVarData(v).vDispatch).QueryInterface(IID, Intf) = S_OK);
  2740. varDispatch or varByRef:
  2741. Result := Assigned(TVarData(v).vPointer) and Assigned(pointer(TVarData(v).vPointer^)) and (IInterface(TVarData(v).vPointer^).QueryInterface(IID, Intf) = S_OK);
  2742. varVariant, varVariant or varByRef:
  2743. Result := Assigned(TVarData(v).vPointer) and VarSupports(Variant(PVarData(TVarData(v).vPointer)^), IID, Intf);
  2744. else
  2745. Result := False;
  2746. end;
  2747. end;
  2748. function VarSupports(const V: Variant; const IID: TGUID): Boolean;
  2749. var
  2750. Dummy: IInterface;
  2751. begin
  2752. Result := VarSupports(V, IID, Dummy);
  2753. end;
  2754. { Variant copy support }
  2755. {$warnings off}
  2756. procedure VarCopyNoInd(var Dest: Variant; const Source: Variant);
  2757. begin
  2758. NotSupported('VarCopyNoInd');
  2759. end;
  2760. {$warnings on}
  2761. {****************************************************************************
  2762. Variant array support procedures and functions
  2763. ****************************************************************************}
  2764. {$r-}
  2765. function VarArrayCreate(const Bounds: array of SizeInt; aVarType: TVarType): Variant;
  2766. var
  2767. hp : PVarArrayBoundArray;
  2768. p : pvararray;
  2769. i,lengthb : SizeInt;
  2770. begin
  2771. if not(VarTypeIsValidArrayType(aVarType)) or odd(length(Bounds)) then
  2772. VarArrayCreateError;
  2773. lengthb:=length(Bounds) div 2;
  2774. try
  2775. GetMem(hp,lengthb*SizeOf(TVarArrayBound));
  2776. for i:=0 to lengthb-1 do
  2777. begin
  2778. hp^[i].LowBound:=Bounds[i*2];
  2779. hp^[i].ElementCount:=Bounds[i*2+1]-Bounds[i*2]+1;
  2780. end;
  2781. SysVarClear(Result);
  2782. p:=SafeArrayCreate(aVarType,lengthb,hp^);
  2783. if not(assigned(p)) then
  2784. VarArrayCreateError;
  2785. TVarData(Result).vType:=aVarType or varArray;
  2786. TVarData(Result).vArray:=p;
  2787. finally
  2788. FreeMem(hp);
  2789. end;
  2790. end;
  2791. {$ifndef RANGECHECKINGOFF}
  2792. {$r+}
  2793. {$endif}
  2794. function VarArrayCreate(const Bounds: PVarArrayBoundArray; Dims : SizeInt; aVarType: TVarType): Variant;
  2795. var
  2796. p : pvararray;
  2797. begin
  2798. if not(VarTypeIsValidArrayType(aVarType)) then
  2799. VarArrayCreateError;
  2800. SysVarClear(Result);
  2801. p:=SafeArrayCreate(aVarType,Dims,Bounds^);
  2802. if not(assigned(p)) then
  2803. VarArrayCreateError;
  2804. TVarData(Result).vType:=aVarType or varArray;
  2805. TVarData(Result).vArray:=p;
  2806. end;
  2807. function VarArrayOf(const Values: array of Variant): Variant;
  2808. var
  2809. i : SizeInt;
  2810. begin
  2811. Result:=VarArrayCreate([0,high(Values)],varVariant);
  2812. for i:=0 to high(Values) do
  2813. Result[i]:=Values[i];
  2814. end;
  2815. function VarArrayAsPSafeArray(const A: Variant): PVarArray;
  2816. var
  2817. v : TVarData;
  2818. begin
  2819. v:=TVarData(a);
  2820. while v.vType=varByRef or varVariant do
  2821. v:=TVarData(v.vPointer^);
  2822. if (v.vType and varArray)=varArray then
  2823. begin
  2824. if (v.vType and varByRef)<>0 then
  2825. Result:=pvararray(v.vPointer^)
  2826. else
  2827. Result:=v.vArray;
  2828. end
  2829. else
  2830. VarResultCheck(VAR_INVALIDARG);
  2831. end;
  2832. function VarArrayDimCount(const A: Variant) : LongInt;
  2833. var
  2834. hv : TVarData;
  2835. begin
  2836. hv:=TVarData(a);
  2837. { get final Variant }
  2838. while hv.vType=varByRef or varVariant do
  2839. hv:=TVarData(hv.vPointer^);
  2840. if (hv.vType and varArray)<>0 then
  2841. Result:=hv.vArray^.DimCount
  2842. else
  2843. Result:=0;
  2844. end;
  2845. function VarArrayLowBound(const A: Variant; Dim: LongInt) : LongInt;
  2846. begin
  2847. VarResultCheck(SafeArrayGetLBound(VarArrayAsPSafeArray(A),Dim,Result));
  2848. end;
  2849. function VarArrayHighBound(const A: Variant; Dim: LongInt) : LongInt;
  2850. begin
  2851. VarResultCheck(SafeArrayGetUBound(VarArrayAsPSafeArray(A),Dim,Result));
  2852. end;
  2853. function VarArrayLock(const A: Variant): Pointer;
  2854. begin
  2855. VarResultCheck(SafeArrayAccessData(VarArrayAsPSafeArray(A),Result));
  2856. end;
  2857. procedure VarArrayUnlock(const A: Variant);
  2858. begin
  2859. VarResultCheck(SafeArrayUnaccessData(VarArrayAsPSafeArray(A)));
  2860. end;
  2861. function VarArrayRef(const A: Variant): Variant;
  2862. begin
  2863. if (TVarData(a).vType and varArray)=0 then
  2864. VarInvalidArgError(TVarData(a).vType);
  2865. TVarData(Result).vType:=TVarData(a).vType or varByRef;
  2866. if (TVarData(a).vType and varByRef)=0 then
  2867. TVarData(Result).vPointer:=@TVarData(a).vArray
  2868. else
  2869. TVarData(Result).vPointer:=@TVarData(a).vPointer;
  2870. end;
  2871. function VarIsArray(const A: Variant; AResolveByRef: Boolean): Boolean;
  2872. var
  2873. v : TVarData;
  2874. begin
  2875. v:=TVarData(a);
  2876. if AResolveByRef then
  2877. while v.vType=varByRef or varVariant do
  2878. v:=TVarData(v.vPointer^);
  2879. Result:=(v.vType and varArray)=varArray;
  2880. end;
  2881. function VarIsArray(const A: Variant): Boolean;
  2882. begin
  2883. VarIsArray:=VarIsArray(A,true);
  2884. end;
  2885. function VarTypeIsValidArrayType(const aVarType: TVarType): Boolean;
  2886. begin
  2887. Result:=aVarType in [varSmallInt,varInteger,
  2888. {$ifndef FPUNONE}
  2889. varSingle,varDouble,varDate,
  2890. {$endif}
  2891. varCurrency,varOleStr,varDispatch,varError,varBoolean,
  2892. varVariant,varUnknown,varShortInt,varByte,varWord,varLongWord];
  2893. end;
  2894. function VarTypeIsValidElementType(const aVarType: TVarType): Boolean;
  2895. var
  2896. customvarianttype : TCustomVariantType;
  2897. begin
  2898. if FindCustomVariantType(aVarType,customvarianttype) then
  2899. Result:=true
  2900. else
  2901. begin
  2902. Result:=(aVarType and not(varByRef) and not(varArray)) in [varEmpty,varNull,varSmallInt,varInteger,
  2903. {$ifndef FPUNONE}
  2904. varSingle,varDouble,varDate,
  2905. {$endif}
  2906. varCurrency,varOleStr,varDispatch,varError,varBoolean,
  2907. varVariant,varUnknown,varShortInt,varByte,varWord,varLongWord,varInt64];
  2908. end;
  2909. end;
  2910. { ---------------------------------------------------------------------
  2911. Variant <-> Dynamic arrays support
  2912. ---------------------------------------------------------------------}
  2913. function DynArrayGetVariantInfo(p : Pointer; var Dims : sizeint) : sizeint;
  2914. begin
  2915. Result:=varNull;
  2916. { skip kind and name }
  2917. inc(Pointer(p),ord(pdynarraytypeinfo(p)^.namelen)+2);
  2918. p:=AlignToPtr(p);
  2919. { skip elesize }
  2920. inc(p,SizeOf(sizeint));
  2921. { search recursive? }
  2922. if pdynarraytypeinfo(ppointer(p)^)^.kind=21{tkDynArr} then
  2923. Result:=DynArrayGetVariantInfo(ppointer(p)^,Dims)
  2924. else
  2925. begin
  2926. { skip dynarraytypeinfo }
  2927. inc(p,SizeOf(pdynarraytypeinfo));
  2928. Result:=plongint(p)^;
  2929. end;
  2930. inc(Dims);
  2931. end;
  2932. {$r-}
  2933. procedure DynArrayToVariant(var V: Variant; const DynArray: Pointer; TypeInfo: Pointer);
  2934. var
  2935. i,
  2936. Dims : sizeint;
  2937. vararrtype,
  2938. dynarrvartype : LongInt;
  2939. vararraybounds : PVarArrayBoundArray;
  2940. iter : TVariantArrayIterator;
  2941. dynarriter : tdynarrayiter;
  2942. p : Pointer;
  2943. temp : Variant;
  2944. variantmanager : tvariantmanager;
  2945. dynarraybounds : tdynarraybounds;
  2946. type
  2947. TDynArray = array of Pointer;
  2948. begin
  2949. DoVarClear(TVarData(v));
  2950. Dims:=0;
  2951. dynarrvartype:=DynArrayGetVariantInfo(TypeInfo,Dims);
  2952. vararrtype:=dynarrvartype;
  2953. if (Dims>1) and not(DynamicArrayIsRectangular(DynArray,TypeInfo)) then
  2954. exit;
  2955. GetVariantManager(variantmanager);
  2956. { retrieve Bounds array }
  2957. Setlength(dynarraybounds,Dims);
  2958. GetMem(vararraybounds,Dims*SizeOf(TVarArrayBound));
  2959. try
  2960. p:=DynArray;
  2961. for i:=0 to Dims-1 do
  2962. begin
  2963. vararraybounds^[i].LowBound:=0;
  2964. vararraybounds^[i].ElementCount:=length(TDynArray(p));
  2965. dynarraybounds[i]:=length(TDynArray(p));
  2966. if dynarraybounds[i]>0 then
  2967. { we checked that the array is rectangular }
  2968. p:=TDynArray(p)[0];
  2969. end;
  2970. { .. create Variant array }
  2971. V:=VarArrayCreate(vararraybounds,Dims,vararrtype);
  2972. VarArrayLock(V);
  2973. try
  2974. iter.init(Dims,PVarArrayBoundArray(vararraybounds));
  2975. dynarriter.init(DynArray,TypeInfo,Dims,dynarraybounds);
  2976. if not iter.AtEnd then
  2977. repeat
  2978. case vararrtype of
  2979. varSmallInt:
  2980. temp:=PSmallInt(dynarriter.data)^;
  2981. varInteger:
  2982. temp:=PInteger(dynarriter.data)^;
  2983. {$ifndef FPUNONE}
  2984. varSingle:
  2985. temp:=PSingle(dynarriter.data)^;
  2986. varDouble:
  2987. temp:=PDouble(dynarriter.data)^;
  2988. varDate:
  2989. temp:=PDouble(dynarriter.data)^;
  2990. {$endif}
  2991. varCurrency:
  2992. temp:=PCurrency(dynarriter.data)^;
  2993. varOleStr:
  2994. temp:=PWideString(dynarriter.data)^;
  2995. varDispatch:
  2996. temp:=PDispatch(dynarriter.data)^;
  2997. varError:
  2998. temp:=PError(dynarriter.data)^;
  2999. varBoolean:
  3000. temp:=PBoolean(dynarriter.data)^;
  3001. varVariant:
  3002. temp:=PVariant(dynarriter.data)^;
  3003. varUnknown:
  3004. temp:=PUnknown(dynarriter.data)^;
  3005. varShortInt:
  3006. temp:=PShortInt(dynarriter.data)^;
  3007. varByte:
  3008. temp:=PByte(dynarriter.data)^;
  3009. varWord:
  3010. temp:=PWord(dynarriter.data)^;
  3011. varLongWord:
  3012. temp:=PLongWord(dynarriter.data)^;
  3013. varInt64:
  3014. temp:=PInt64(dynarriter.data)^;
  3015. varQWord:
  3016. temp:=PQWord(dynarriter.data)^;
  3017. else
  3018. VarClear(temp);
  3019. end;
  3020. dynarriter.next;
  3021. variantmanager.VarArrayPut(V,temp,Dims,PLongint(iter.Coords));
  3022. until not(iter.next);
  3023. finally
  3024. iter.done;
  3025. dynarriter.done;
  3026. VarArrayUnlock(V);
  3027. end;
  3028. finally
  3029. FreeMem(vararraybounds);
  3030. end;
  3031. end;
  3032. procedure DynArrayFromVariant(var DynArray: Pointer; const V: Variant; TypeInfo: Pointer);
  3033. var
  3034. DynArrayDims,
  3035. VarArrayDims : SizeInt;
  3036. iter : TVariantArrayIterator;
  3037. dynarriter : tdynarrayiter;
  3038. temp : Variant;
  3039. dynarrvartype : LongInt;
  3040. variantmanager : tvariantmanager;
  3041. vararraybounds : PVarArrayBoundArray;
  3042. dynarraybounds : tdynarraybounds;
  3043. i : SizeInt;
  3044. type
  3045. TDynArray = array of Pointer;
  3046. begin
  3047. VarArrayDims:=VarArrayDimCount(V);
  3048. DynArrayDims:=0;
  3049. dynarrvartype:=DynArrayGetVariantInfo(TypeInfo,DynArrayDims);
  3050. if (VarArrayDims=0) or (VarArrayDims<>DynArrayDims) then
  3051. VarResultCheck(VAR_INVALIDARG);
  3052. { retrieve Bounds array }
  3053. Setlength(dynarraybounds,VarArrayDims);
  3054. GetMem(vararraybounds,VarArrayDims*SizeOf(TVarArrayBound));
  3055. try
  3056. for i:=0 to VarArrayDims-1 do
  3057. begin
  3058. vararraybounds^[i].LowBound:=VarArrayLowBound(V,i+1);
  3059. vararraybounds^[i].ElementCount:=VarArrayHighBound(V,i+1)-vararraybounds^[i].LowBound+1;
  3060. dynarraybounds[i]:=vararraybounds^[i].ElementCount;
  3061. end;
  3062. DynArraySetLength(DynArray,TypeInfo,VarArrayDims,PSizeInt(dynarraybounds));
  3063. GetVariantManager(variantmanager);
  3064. VarArrayLock(V);
  3065. try
  3066. iter.init(VarArrayDims,PVarArrayBoundArray(vararraybounds));
  3067. dynarriter.init(DynArray,TypeInfo,VarArrayDims,dynarraybounds);
  3068. if not iter.AtEnd then
  3069. repeat
  3070. temp:=variantmanager.VarArrayGet(V,VarArrayDims,PLongint(iter.Coords));
  3071. case dynarrvartype of
  3072. varSmallInt:
  3073. PSmallInt(dynarriter.data)^:=temp;
  3074. varInteger:
  3075. PInteger(dynarriter.data)^:=temp;
  3076. {$ifndef FPUNONE}
  3077. varSingle:
  3078. PSingle(dynarriter.data)^:=temp;
  3079. varDouble:
  3080. PDouble(dynarriter.data)^:=temp;
  3081. varDate:
  3082. PDouble(dynarriter.data)^:=temp;
  3083. {$endif}
  3084. varCurrency:
  3085. PCurrency(dynarriter.data)^:=temp;
  3086. varOleStr:
  3087. PWideString(dynarriter.data)^:=temp;
  3088. varDispatch:
  3089. PDispatch(dynarriter.data)^:=temp;
  3090. varError:
  3091. PError(dynarriter.data)^:=temp;
  3092. varBoolean:
  3093. PBoolean(dynarriter.data)^:=temp;
  3094. varVariant:
  3095. PVariant(dynarriter.data)^:=temp;
  3096. varUnknown:
  3097. PUnknown(dynarriter.data)^:=temp;
  3098. varShortInt:
  3099. PShortInt(dynarriter.data)^:=temp;
  3100. varByte:
  3101. PByte(dynarriter.data)^:=temp;
  3102. varWord:
  3103. PWord(dynarriter.data)^:=temp;
  3104. varLongWord:
  3105. PLongWord(dynarriter.data)^:=temp;
  3106. varInt64:
  3107. PInt64(dynarriter.data)^:=temp;
  3108. varQWord:
  3109. PQWord(dynarriter.data)^:=temp;
  3110. else
  3111. VarCastError;
  3112. end;
  3113. dynarriter.next;
  3114. until not(iter.next);
  3115. finally
  3116. iter.done;
  3117. dynarriter.done;
  3118. VarArrayUnlock(V);
  3119. end;
  3120. finally
  3121. FreeMem(vararraybounds);
  3122. end;
  3123. end;
  3124. {$ifndef RANGECHECKINGOFF}
  3125. {$r+}
  3126. {$endif}
  3127. function FindCustomVariantType(const aVarType: TVarType; out CustomVariantType: TCustomVariantType): Boolean; overload;
  3128. begin
  3129. Result:=(aVarType>=CMinVarType);
  3130. if Result then
  3131. begin
  3132. EnterCriticalSection(customvarianttypelock);
  3133. try
  3134. Result:=(aVarType-CMinVarType)<=high(customvarianttypes);
  3135. if Result then
  3136. begin
  3137. CustomVariantType:=customvarianttypes[aVarType-CMinVarType];
  3138. Result:=assigned(CustomVariantType) and
  3139. (CustomVariantType<>InvalidCustomVariantType);
  3140. end;
  3141. finally
  3142. LeaveCriticalSection(customvarianttypelock);
  3143. end;
  3144. end;
  3145. end;
  3146. function FindCustomVariantType(const TypeName: string; out CustomVariantType: TCustomVariantType): Boolean; overload;
  3147. var
  3148. i: Integer;
  3149. tmp: TCustomVariantType;
  3150. ShortTypeName: shortstring;
  3151. begin
  3152. ShortTypeName:=TypeName; // avoid conversion in the loop
  3153. result:=False;
  3154. EnterCriticalSection(customvarianttypelock);
  3155. try
  3156. for i:=low(customvarianttypes) to high(customvarianttypes) do
  3157. begin
  3158. tmp:=customvarianttypes[i];
  3159. result:=Assigned(tmp) and (tmp<>InvalidCustomVariantType) and
  3160. tmp.ClassNameIs(ShortTypeName);
  3161. if result then
  3162. begin
  3163. CustomVariantType:=tmp;
  3164. Exit;
  3165. end;
  3166. end;
  3167. finally
  3168. LeaveCriticalSection(customvarianttypelock);
  3169. end;
  3170. end;
  3171. function Unassigned: Variant; // Unassigned standard constant
  3172. begin
  3173. SysVarClear(Result);
  3174. TVarData(Result).vType := varEmpty;
  3175. end;
  3176. function Null: Variant; // Null standard constant
  3177. begin
  3178. SysVarClear(Result);
  3179. TVarData(Result).vType := varNull;
  3180. end;
  3181. procedure VarDispInvokeError;
  3182. begin
  3183. raise EVariantDispatchError.Create(SDispatchError);
  3184. end;
  3185. { ---------------------------------------------------------------------
  3186. TCustomVariantType Class.
  3187. ---------------------------------------------------------------------}
  3188. { All TCustomVariantType descendants are singletons, they ignore automatic refcounting. }
  3189. function TCustomVariantType.QueryInterface(const IID: TGUID; out Obj): HResult; stdcall;
  3190. begin
  3191. if GetInterface(IID, obj) then
  3192. result := S_OK
  3193. else
  3194. result := E_NOINTERFACE;
  3195. end;
  3196. function TCustomVariantType._AddRef: Integer; stdcall;
  3197. begin
  3198. result := -1;
  3199. end;
  3200. function TCustomVariantType._Release: Integer; stdcall;
  3201. begin
  3202. result := -1;
  3203. end;
  3204. {$warnings off}
  3205. procedure TCustomVariantType.SimplisticClear(var V: TVarData);
  3206. begin
  3207. NotSupported('TCustomVariantType.SimplisticClear');
  3208. end;
  3209. procedure TCustomVariantType.SimplisticCopy(var Dest: TVarData; const Source: TVarData; const Indirect: Boolean = False);
  3210. begin
  3211. NotSupported('TCustomVariantType.SimplisticCopy');
  3212. end;
  3213. procedure TCustomVariantType.RaiseInvalidOp;
  3214. begin
  3215. VarInvalidOp;
  3216. end;
  3217. procedure TCustomVariantType.RaiseCastError;
  3218. begin
  3219. VarCastError;
  3220. end;
  3221. procedure TCustomVariantType.RaiseDispError;
  3222. begin
  3223. VarDispInvokeError;
  3224. end;
  3225. function TCustomVariantType.LeftPromotion(const V: TVarData; const Operation: TVarOp; out RequiredVarType: TVarType): Boolean;
  3226. begin
  3227. NotSupported('TCustomVariantType.LeftPromotion');
  3228. end;
  3229. function TCustomVariantType.RightPromotion(const V: TVarData; const Operation: TVarOp; out RequiredVarType: TVarType): Boolean;
  3230. begin
  3231. NotSupported('TCustomVariantType.RightPromotion');
  3232. end;
  3233. function TCustomVariantType.OlePromotion(const V: TVarData; out RequiredVarType: TVarType): Boolean;
  3234. begin
  3235. NotSupported('TCustomVariantType.OlePromotion');
  3236. end;
  3237. procedure TCustomVariantType.DispInvoke(Dest: PVarData; const Source: TVarData; CallDesc: PCallDesc; Params: Pointer);
  3238. begin
  3239. RaiseDispError;
  3240. end;
  3241. procedure TCustomVariantType.VarDataInit(var Dest: TVarData);
  3242. begin
  3243. FillChar(Dest,SizeOf(Dest),0);
  3244. end;
  3245. procedure TCustomVariantType.VarDataClear(var Dest: TVarData);
  3246. begin
  3247. VarClearProc(Dest);
  3248. end;
  3249. procedure TCustomVariantType.VarDataCopy(var Dest: TVarData; const Source: TVarData);
  3250. begin
  3251. DoVarCopy(Dest,Source)
  3252. end;
  3253. procedure TCustomVariantType.VarDataCopyNoInd(var Dest: TVarData; const Source: TVarData);
  3254. begin
  3255. // This is probably not correct, but there is no DoVarCopyInd
  3256. DoVarCopy(Dest,Source);
  3257. end;
  3258. procedure TCustomVariantType.VarDataCast(var Dest: TVarData; const Source: TVarData);
  3259. begin
  3260. DoVarCast(Dest, Source, VarType);
  3261. end;
  3262. procedure TCustomVariantType.VarDataCastTo(var Dest: TVarData; const Source: TVarData; const aVarType: TVarType);
  3263. begin
  3264. DoVarCast(Dest, Source, AVarType);
  3265. end;
  3266. procedure TCustomVariantType.VarDataCastTo(var Dest: TVarData; const aVarType: TVarType);
  3267. begin
  3268. DoVarCast(Dest,Dest,AVarType);
  3269. end;
  3270. procedure TCustomVariantType.VarDataCastToOleStr(var Dest: TVarData);
  3271. begin
  3272. VarDataCastTo(Dest, Dest, varOleStr);
  3273. end;
  3274. procedure TCustomVariantType.VarDataFromStr(var V: TVarData; const Value: string);
  3275. begin
  3276. sysvarfromlstr(Variant(V),Value);
  3277. end;
  3278. procedure TCustomVariantType.VarDataFromOleStr(var V: TVarData; const Value: WideString);
  3279. begin
  3280. sysvarfromwstr(variant(V),Value);
  3281. end;
  3282. function TCustomVariantType.VarDataToStr(const V: TVarData): string;
  3283. begin
  3284. sysvartolstr(Result,Variant(V));
  3285. end;
  3286. function TCustomVariantType.VarDataIsEmptyParam(const V: TVarData): Boolean;
  3287. begin
  3288. VarIsEmptyParam(Variant(V));
  3289. end;
  3290. function TCustomVariantType.VarDataIsByRef(const V: TVarData): Boolean;
  3291. begin
  3292. Result:=(V.vType and varByRef)=varByRef;
  3293. end;
  3294. function TCustomVariantType.VarDataIsArray(const V: TVarData): Boolean;
  3295. begin
  3296. Result:=(V.vType and varArray)=varArray;
  3297. end;
  3298. function TCustomVariantType.VarDataIsOrdinal(const V: TVarData): Boolean;
  3299. begin
  3300. Result:=(V.vType and varTypeMask) in OrdinalVarTypes;
  3301. end;
  3302. function TCustomVariantType.VarDataIsFloat(const V: TVarData): Boolean;
  3303. begin
  3304. Result:=(V.vType and varTypeMask) in FloatVarTypes;
  3305. end;
  3306. function TCustomVariantType.VarDataIsNumeric(const V: TVarData): Boolean;
  3307. begin
  3308. Result:=(V.vType and varTypeMask) in (OrdinalVarTypes + FloatVarTypes);
  3309. end;
  3310. function TCustomVariantType.VarDataIsStr(const V: TVarData): Boolean;
  3311. begin
  3312. Result:=
  3313. ((V.vType and varTypeMask) = varOleStr) or
  3314. ((V.vType and varTypeMask) = varString);
  3315. end;
  3316. procedure RegisterCustomVariantType(obj: TCustomVariantType; RequestedVarType: TVarType;
  3317. UseFirstAvailable: Boolean);
  3318. var
  3319. index,L: Integer;
  3320. begin
  3321. EnterCriticalSection(customvarianttypelock);
  3322. try
  3323. L:=Length(customvarianttypes);
  3324. if UseFirstAvailable then
  3325. begin
  3326. repeat
  3327. inc(customvariantcurrtype);
  3328. if customvariantcurrtype>=CMaxVarType then
  3329. raise EVariantError.Create(SVarTypeTooManyCustom);
  3330. until ((customvariantcurrtype-CMinVarType)>=L) or
  3331. (customvarianttypes[customvariantcurrtype-CMinVarType]=nil);
  3332. RequestedVarType:=customvariantcurrtype;
  3333. end
  3334. else if (RequestedVarType<CFirstUserType) or (RequestedVarType>CMaxVarType) then
  3335. raise EVariantError.CreateFmt(SVarTypeOutOfRangeWithPrefix, ['$', RequestedVarType]);
  3336. index:=RequestedVarType-CMinVarType;
  3337. if index>=L then
  3338. SetLength(customvarianttypes,L+1);
  3339. if Assigned(customvarianttypes[index]) then
  3340. begin
  3341. if customvarianttypes[index]=InvalidCustomVariantType then
  3342. raise EVariantError.CreateFmt(SVarTypeNotUsableWithPrefix, ['$', RequestedVarType])
  3343. else
  3344. raise EVariantError.CreateFmt(SVarTypeAlreadyUsedWithPrefix,
  3345. ['$', RequestedVarType, customvarianttypes[index].ClassName]);
  3346. end;
  3347. customvarianttypes[index]:=obj;
  3348. obj.FVarType:=RequestedVarType;
  3349. finally
  3350. LeaveCriticalSection(customvarianttypelock);
  3351. end;
  3352. end;
  3353. constructor TCustomVariantType.Create;
  3354. begin
  3355. RegisterCustomVariantType(Self,0,True);
  3356. end;
  3357. constructor TCustomVariantType.Create(RequestedVarType: TVarType);
  3358. begin
  3359. RegisterCustomVariantType(Self,RequestedVarType,False);
  3360. end;
  3361. destructor TCustomVariantType.Destroy;
  3362. begin
  3363. EnterCriticalSection(customvarianttypelock);
  3364. try
  3365. if FVarType<>0 then
  3366. customvarianttypes[FVarType-CMinVarType]:=InvalidCustomVariantType;
  3367. finally
  3368. LeaveCriticalSection(customvarianttypelock);
  3369. end;
  3370. inherited Destroy;
  3371. end;
  3372. function TCustomVariantType.IsClear(const V: TVarData): Boolean;
  3373. Var
  3374. VT : TVarType;
  3375. begin
  3376. VT:=V.vType and varTypeMask;
  3377. Result:=(VT=varEmpty) or (((VT=varDispatch) or (VT=varUnknown))
  3378. and (TVarData(V).vDispatch=Nil));
  3379. end;
  3380. procedure TCustomVariantType.Cast(var Dest: TVarData; const Source: TVarData);
  3381. begin
  3382. DoVarCast(Dest,Source,VarType);
  3383. end;
  3384. procedure TCustomVariantType.CastTo(var Dest: TVarData; const Source: TVarData; const aVarType: TVarType);
  3385. begin
  3386. DoVarCast(Dest,Source,AVarType);
  3387. end;
  3388. procedure TCustomVariantType.CastToOle(var Dest: TVarData; const Source: TVarData);
  3389. begin
  3390. NotSupported('TCustomVariantType.CastToOle');
  3391. end;
  3392. procedure TCustomVariantType.BinaryOp(var Left: TVarData; const Right: TVarData; const Operation: TVarOp);
  3393. begin
  3394. NotSupported('TCustomVariantType.BinaryOp');
  3395. end;
  3396. procedure TCustomVariantType.UnaryOp(var Right: TVarData; const Operation: TVarOp);
  3397. begin
  3398. NotSupported('TCustomVariantType.UnaryOp');
  3399. end;
  3400. function TCustomVariantType.CompareOp(const Left, Right: TVarData; const Operation: TVarOp): Boolean;
  3401. begin
  3402. NotSupported('TCustomVariantType.CompareOp');
  3403. end;
  3404. procedure TCustomVariantType.Compare(const Left, Right: TVarData; var Relationship: TVarCompareResult);
  3405. begin
  3406. NotSupported('TCustomVariantType.Compare');
  3407. end;
  3408. {$warnings on}
  3409. { ---------------------------------------------------------------------
  3410. TInvokeableVariantType implementation
  3411. ---------------------------------------------------------------------}
  3412. procedure TInvokeableVariantType.DispInvoke(Dest: PVarData; const Source: TVarData; CallDesc: PCallDesc; Params: Pointer);
  3413. begin
  3414. NotSupported('TInvokeableVariantType.DispInvoke');
  3415. end;
  3416. function TInvokeableVariantType.DoFunction(var Dest: TVarData; const V: TVarData; const Name: string; const Arguments: TVarDataArray): Boolean;
  3417. begin
  3418. result := False;
  3419. end;
  3420. function TInvokeableVariantType.DoProcedure(const V: TVarData; const Name: string; const Arguments: TVarDataArray): Boolean;
  3421. begin
  3422. result := False
  3423. end;
  3424. function TInvokeableVariantType.GetProperty(var Dest: TVarData; const V: TVarData; const Name: string): Boolean;
  3425. begin
  3426. result := False;
  3427. end;
  3428. function TInvokeableVariantType.SetProperty(const V: TVarData; const Name: string; const Value: TVarData): Boolean;
  3429. begin
  3430. result := False;
  3431. end;
  3432. { ---------------------------------------------------------------------
  3433. TPublishableVariantType implementation
  3434. ---------------------------------------------------------------------}
  3435. function TPublishableVariantType.GetProperty(var Dest: TVarData; const V: TVarData; const Name: string): Boolean;
  3436. begin
  3437. Result:=true;
  3438. Variant(Dest):=GetPropValue(getinstance(v),name);
  3439. end;
  3440. function TPublishableVariantType.SetProperty(const V: TVarData; const Name: string; const Value: TVarData): Boolean;
  3441. begin
  3442. Result:=true;
  3443. SetPropValue(getinstance(v),name,Variant(value));
  3444. end;
  3445. procedure VarCastError;
  3446. begin
  3447. raise EVariantTypeCastError.Create(SInvalidVarCast);
  3448. end;
  3449. procedure VarCastError(const ASourceType, ADestType: TVarType);
  3450. begin
  3451. raise EVariantTypeCastError.CreateFmt(SVarTypeCouldNotConvert,
  3452. [VarTypeAsText(ASourceType),VarTypeAsText(ADestType)]);
  3453. end;
  3454. procedure VarCastErrorOle(const ASourceType: TVarType);
  3455. begin
  3456. raise EVariantTypeCastError.CreateFmt(SVarTypeCouldNotConvert,
  3457. [VarTypeAsText(ASourceType),'(OleVariant)']);
  3458. end;
  3459. procedure VarInvalidOp;
  3460. begin
  3461. raise EVariantInvalidOpError.Create(SInvalidVarOp);
  3462. end;
  3463. procedure VarInvalidOp(const aLeft, aRight: TVarType; aOpCode: TVarOp);
  3464. begin
  3465. raise EVariantInvalidOpError.CreateFmt(SInvalidBinaryVarOp,
  3466. [VarTypeAsText(aLeft),VarOpAsText[aOpCode],VarTypeAsText(aRight)]);
  3467. end;
  3468. procedure VarInvalidOp(const aRight: TVarType; aOpCode: TVarOp);
  3469. begin
  3470. raise EVariantInvalidOpError.CreateFmt(SInvalidUnaryVarOp,
  3471. [VarOpAsText[aOpCode],VarTypeAsText(aRight)]);
  3472. end;
  3473. procedure VarInvalidNullOp;
  3474. begin
  3475. raise EVariantInvalidOpError.Create(SInvalidvarNullOp);
  3476. end;
  3477. procedure VarParamNotFoundError;
  3478. begin
  3479. raise EVariantParamNotFoundError.Create(SVarParamNotFound);
  3480. end;
  3481. procedure VarBadTypeError;
  3482. begin
  3483. raise EVariantBadVarTypeError.Create(SVarBadType);
  3484. end;
  3485. procedure VarOverflowError;
  3486. begin
  3487. raise EVariantOverflowError.Create(SVarOverflow);
  3488. end;
  3489. procedure VarOverflowError(const ASourceType, ADestType: TVarType);
  3490. begin
  3491. raise EVariantOverflowError.CreateFmt(SVarTypeConvertOverflow,
  3492. [VarTypeAsText(ASourceType),VarTypeAsText(ADestType)]);
  3493. end;
  3494. procedure VarRangeCheckError(const AType: TVarType);
  3495. begin
  3496. raise EVariantOverflowError.CreateFmt(SVarTypeRangeCheck1,
  3497. [VarTypeAsText(AType)])
  3498. end;
  3499. procedure VarRangeCheckError(const ASourceType, ADestType: TVarType);
  3500. begin
  3501. if ASourceType<>ADestType then
  3502. raise EVariantOverflowError.CreateFmt(SVarTypeRangeCheck2,
  3503. [VarTypeAsText(ASourceType),VarTypeAsText(ADestType)])
  3504. else
  3505. VarRangeCheckError(ASourceType);
  3506. end;
  3507. procedure VarBadIndexError;
  3508. begin
  3509. raise EVariantBadIndexError.Create(SVarArrayBounds);
  3510. end;
  3511. procedure VarArrayLockedError;
  3512. begin
  3513. raise EVariantArrayLockedError.Create(SVarArrayLocked);
  3514. end;
  3515. procedure VarNotImplError;
  3516. begin
  3517. raise EVariantNotImplError.Create(SVarNotImplemented);
  3518. end;
  3519. procedure VarOutOfMemoryError;
  3520. begin
  3521. raise EVariantOutOfMemoryError.Create(SOutOfMemory);
  3522. end;
  3523. procedure VarInvalidArgError;
  3524. begin
  3525. raise EVariantInvalidArgError.Create(SVarInvalid);
  3526. end;
  3527. procedure VarInvalidArgError(AType: TVarType);
  3528. begin
  3529. raise EVariantInvalidArgError.CreateFmt(SVarInvalid1,
  3530. [VarTypeAsText(AType)])
  3531. end;
  3532. procedure VarUnexpectedError;
  3533. begin
  3534. raise EVariantUnexpectedError.Create(SVarUnexpected);
  3535. end;
  3536. procedure VarArrayCreateError;
  3537. begin
  3538. raise EVariantArrayCreateError.Create(SVarArrayCreate);
  3539. end;
  3540. procedure RaiseVarException(res : HRESULT);
  3541. begin
  3542. case res of
  3543. VAR_PARAMNOTFOUND:
  3544. VarParamNotFoundError;
  3545. VAR_TYPEMISMATCH:
  3546. VarCastError;
  3547. VAR_BADVARTYPE:
  3548. VarBadTypeError;
  3549. VAR_EXCEPTION:
  3550. VarInvalidOp;
  3551. VAR_OVERFLOW:
  3552. VarOverflowError;
  3553. VAR_BADINDEX:
  3554. VarBadIndexError;
  3555. VAR_ARRAYISLOCKED:
  3556. VarArrayLockedError;
  3557. VAR_NOTIMPL:
  3558. VarNotImplError;
  3559. VAR_OUTOFMEMORY:
  3560. VarOutOfMemoryError;
  3561. VAR_INVALIDARG:
  3562. VarInvalidArgError;
  3563. VAR_UNEXPECTED:
  3564. VarUnexpectedError;
  3565. else
  3566. raise EVariantError.CreateFmt(SInvalidVarOpWithHResultWithPrefix,
  3567. ['$',res,'']);
  3568. end;
  3569. end;
  3570. procedure VarResultCheck(AResult: HRESULT);{$IFDEF VARIANTINLINE}inline;{$ENDIF VARIANTINLINE}
  3571. begin
  3572. if AResult<>VAR_OK then
  3573. RaiseVarException(AResult);
  3574. end;
  3575. procedure VarResultCheck(AResult: HRESULT; ASourceType, ADestType: TVarType);
  3576. begin
  3577. case AResult of
  3578. VAR_OK:
  3579. ;
  3580. VAR_OVERFLOW:
  3581. VarOverflowError(ASourceType,ADestType);
  3582. VAR_TYPEMISMATCH:
  3583. VarCastError(ASourceType,ADestType);
  3584. else
  3585. RaiseVarException(AResult);
  3586. end;
  3587. end;
  3588. procedure HandleConversionException(const ASourceType, ADestType: TVarType);
  3589. begin
  3590. if exceptobject is econverterror then
  3591. VarCastError(asourcetype,adesttype)
  3592. else if (exceptobject is eoverflow) or
  3593. (exceptobject is erangeerror) then
  3594. varoverflowerror(asourcetype,adesttype)
  3595. else
  3596. raise exception(acquireexceptionobject);
  3597. end;
  3598. function VarTypeAsText(const AType: TVarType): string;
  3599. var
  3600. customvarianttype : TCustomVariantType;
  3601. const
  3602. names : array[varEmpty..varQWord] of string[8] = (
  3603. 'Empty','Null','Smallint','Integer','Single','Double','Currency','Date','OleStr','Dispatch','Error','Boolean','Variant',
  3604. 'Unknown','Decimal','???','ShortInt','Byte','Word','DWord','Int64','QWord');
  3605. begin
  3606. if ((AType and varTypeMask)>=low(names)) and ((AType and varTypeMask)<=high(names)) then
  3607. Result:=names[AType and varTypeMask]
  3608. else
  3609. case AType and varTypeMask of
  3610. varString:
  3611. Result:='String';
  3612. varAny:
  3613. Result:='Any';
  3614. else
  3615. begin
  3616. if FindCustomVariantType(AType and varTypeMask,customvarianttype) then
  3617. Result:=customvarianttype.classname
  3618. else
  3619. Result:='$'+IntToHex(AType and varTypeMask,4)
  3620. end;
  3621. end;
  3622. if (AType and vararray)<>0 then
  3623. Result:='Array of '+Result;
  3624. if (AType and varByRef)<>0 then
  3625. Result:='Ref to '+Result;
  3626. end;
  3627. function FindVarData(const V: Variant): PVarData;
  3628. begin
  3629. Result:=PVarData(@V);
  3630. while Result^.vType=varVariant or varByRef do
  3631. Result:=PVarData(Result^.vPointer);
  3632. end;
  3633. { ---------------------------------------------------------------------
  3634. Variant properties from typinfo
  3635. ---------------------------------------------------------------------}
  3636. function GetVariantProp(Instance : TObject;PropInfo : PPropInfo) : Variant;
  3637. type
  3638. TGetVariantProc = function:Variant of object;
  3639. TGetVariantProcIndex = function(Index: integer): Variant of object;
  3640. var
  3641. AMethod : TMethod;
  3642. begin
  3643. Result:=Null;
  3644. case PropInfo^.PropProcs and 3 of
  3645. ptField:
  3646. Result:=PVariant(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
  3647. ptStatic,
  3648. ptVirtual:
  3649. begin
  3650. if (PropInfo^.PropProcs and 3)=ptStatic then
  3651. AMethod.Code:=PropInfo^.GetProc
  3652. else
  3653. AMethod.Code:=PPointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;
  3654. AMethod.Data:=Instance;
  3655. if ((PropInfo^.PropProcs shr 6) and 1)=0 then
  3656. Result:=TGetVariantProc(AMethod)()
  3657. else
  3658. Result:=TGetVariantProcIndex(AMethod)(PropInfo^.Index);
  3659. end;
  3660. end;
  3661. end;
  3662. Procedure SetVariantProp(Instance : TObject;PropInfo : PPropInfo; const Value : Variant);
  3663. type
  3664. TSetVariantProc = procedure(const AValue: Variant) of object;
  3665. TSetVariantProcIndex = procedure(Index: integer; AValue: Variant) of object;
  3666. Var
  3667. AMethod : TMethod;
  3668. begin
  3669. case (PropInfo^.PropProcs shr 2) and 3 of
  3670. ptfield:
  3671. PVariant(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;
  3672. ptVirtual,ptStatic:
  3673. begin
  3674. if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
  3675. AMethod.Code:=PropInfo^.SetProc
  3676. else
  3677. AMethod.Code:=PPointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
  3678. AMethod.Data:=Instance;
  3679. if ((PropInfo^.PropProcs shr 6) and 1)=0 then
  3680. TSetVariantProc(AMethod)(Value)
  3681. else
  3682. TSetVariantProcIndex(AMethod)(PropInfo^.Index,Value);
  3683. end;
  3684. end;
  3685. end;
  3686. Function GetVariantProp(Instance: TObject; const PropName: string): Variant;
  3687. begin
  3688. Result:=GetVariantProp(Instance,FindPropInfo(Instance,PropName));
  3689. end;
  3690. Procedure SetVariantProp(Instance: TObject; const PropName: string; const Value: Variant);
  3691. begin
  3692. SetVariantprop(instance,FindpropInfo(Instance,PropName),Value);
  3693. end;
  3694. { ---------------------------------------------------------------------
  3695. All properties through Variant.
  3696. ---------------------------------------------------------------------}
  3697. Function GetPropValue(Instance: TObject; const PropName: string): Variant;
  3698. begin
  3699. Result:=GetPropValue(Instance,PropName,True);
  3700. end;
  3701. Function GetPropValue(Instance: TObject; const PropName: string; PreferStrings: Boolean): Variant;
  3702. var
  3703. PropInfo: PPropInfo;
  3704. begin
  3705. // find the property
  3706. PropInfo := GetPropInfo(Instance, PropName);
  3707. if PropInfo = nil then
  3708. raise EPropertyError.CreateFmt(SErrPropertyNotFound, [PropName])
  3709. else
  3710. begin
  3711. Result := Null; //at worst
  3712. // call the Right GetxxxProp
  3713. case PropInfo^.PropType^.Kind of
  3714. tkInteger, tkChar, tkWChar, tkClass, tkBool:
  3715. Result := GetOrdProp(Instance, PropInfo);
  3716. tkEnumeration:
  3717. if PreferStrings then
  3718. Result := GetEnumProp(Instance, PropInfo)
  3719. else
  3720. Result := GetOrdProp(Instance, PropInfo);
  3721. tkSet:
  3722. if PreferStrings then
  3723. Result := GetSetProp(Instance, PropInfo, False)
  3724. else
  3725. Result := GetOrdProp(Instance, PropInfo);
  3726. {$ifndef FPUNONE}
  3727. tkFloat:
  3728. Result := GetFloatProp(Instance, PropInfo);
  3729. {$endif}
  3730. tkMethod:
  3731. Result := PropInfo^.PropType^.Name;
  3732. tkString, tkLString, tkAString:
  3733. Result := GetStrProp(Instance, PropInfo);
  3734. tkWString:
  3735. Result := GetWideStrProp(Instance, PropInfo);
  3736. tkUString:
  3737. Result := GetUnicodeStrProp(Instance, PropInfo);
  3738. tkVariant:
  3739. Result := GetVariantProp(Instance, PropInfo);
  3740. tkInt64:
  3741. Result := GetInt64Prop(Instance, PropInfo);
  3742. tkQWord:
  3743. Result := QWord(GetInt64Prop(Instance, PropInfo));
  3744. else
  3745. raise EPropertyConvertError.CreateFmt('Invalid Property Type: %s',[PropInfo^.PropType^.Name]);
  3746. end;
  3747. end;
  3748. end;
  3749. Procedure SetPropValue(Instance: TObject; const PropName: string; const Value: Variant);
  3750. var
  3751. PropInfo: PPropInfo;
  3752. TypeData: PTypeData;
  3753. O: Integer;
  3754. I64: Int64;
  3755. Qw: QWord;
  3756. S: String;
  3757. B: Boolean;
  3758. begin
  3759. // find the property
  3760. PropInfo := GetPropInfo(Instance, PropName);
  3761. if PropInfo = nil then
  3762. raise EPropertyError.CreateFmt(SErrPropertyNotFound, [PropName])
  3763. else
  3764. begin
  3765. TypeData := GetTypeData(PropInfo^.PropType);
  3766. // call Right SetxxxProp
  3767. case PropInfo^.PropType^.Kind of
  3768. tkBool:
  3769. begin
  3770. { to support the strings 'true' and 'false' }
  3771. if (VarType(Value)=varOleStr) or
  3772. (VarType(Value)=varString) or
  3773. (VarType(Value)=varBoolean) then
  3774. begin
  3775. B:=Value;
  3776. SetOrdProp(Instance, PropInfo, ord(B));
  3777. end
  3778. else
  3779. begin
  3780. I64:=Value;
  3781. if (I64<TypeData^.MinValue) or (I64>TypeData^.MaxValue) then
  3782. raise ERangeError.Create(SRangeError);
  3783. SetOrdProp(Instance, PropInfo, I64);
  3784. end;
  3785. end;
  3786. tkInteger, tkChar, tkWChar:
  3787. begin
  3788. I64:=Value;
  3789. if (TypeData^.OrdType=otULong) then
  3790. if (I64<LongWord(TypeData^.MinValue)) or (I64>LongWord(TypeData^.MaxValue)) then
  3791. raise ERangeError.Create(SRangeError)
  3792. else
  3793. else
  3794. if (I64<TypeData^.MinValue) or (I64>TypeData^.MaxValue) then
  3795. raise ERangeError.Create(SRangeError);
  3796. SetOrdProp(Instance, PropInfo, I64);
  3797. end;
  3798. tkEnumeration :
  3799. begin
  3800. if (VarType(Value)=varOleStr) or (VarType(Value)=varString) then
  3801. begin
  3802. S:=Value;
  3803. SetEnumProp(Instance,PropInfo,S);
  3804. end
  3805. else
  3806. begin
  3807. I64:=Value;
  3808. if (I64<TypeData^.MinValue) or (I64>TypeData^.MaxValue) then
  3809. raise ERangeError.Create(SRangeError);
  3810. SetOrdProp(Instance, PropInfo, I64);
  3811. end;
  3812. end;
  3813. tkSet :
  3814. begin
  3815. if (VarType(Value)=varOleStr) or (VarType(Value)=varString) then
  3816. begin
  3817. S:=Value;
  3818. SetSetProp(Instance,PropInfo,S);
  3819. end
  3820. else
  3821. begin
  3822. O:=Value;
  3823. SetOrdProp(Instance, PropInfo, O);
  3824. end;
  3825. end;
  3826. {$ifndef FPUNONE}
  3827. tkFloat:
  3828. SetFloatProp(Instance, PropInfo, Value);
  3829. {$endif}
  3830. tkString, tkLString, tkAString:
  3831. SetStrProp(Instance, PropInfo, VarToStr(Value));
  3832. tkWString:
  3833. SetWideStrProp(Instance, PropInfo, VarToWideStr(Value));
  3834. tkUString:
  3835. SetUnicodeStrProp(Instance, PropInfo, VarToUnicodeStr(Value));
  3836. tkVariant:
  3837. SetVariantProp(Instance, PropInfo, Value);
  3838. tkInt64:
  3839. begin
  3840. I64:=Value;
  3841. if (I64<TypeData^.MinInt64Value) or (I64>TypeData^.MaxInt64Value) then
  3842. raise ERangeError.Create(SRangeError);
  3843. SetInt64Prop(Instance, PropInfo, I64);
  3844. end;
  3845. tkQWord:
  3846. begin
  3847. Qw:=Value;
  3848. if (Qw<TypeData^.MinQWordValue) or (Qw>TypeData^.MaxQWordValue) then
  3849. raise ERangeError.Create(SRangeError);
  3850. SetInt64Prop(Instance, PropInfo,Qw);
  3851. end
  3852. else
  3853. raise EPropertyConvertError.CreateFmt('SetPropValue: Invalid Property Type %s',
  3854. [PropInfo^.PropType^.Name]);
  3855. end;
  3856. end;
  3857. end;
  3858. var
  3859. i : LongInt;
  3860. Initialization
  3861. InitCriticalSection(customvarianttypelock);
  3862. // start with one-less value, so first increment yields CFirstUserType
  3863. customvariantcurrtype:=CFirstUserType-1;
  3864. SetSysVariantManager;
  3865. SetClearVarToEmptyParam(TVarData(EmptyParam));
  3866. VarClearProc:=@DoVarClear;
  3867. VarAddRefProc:=@DoVarAddRef;
  3868. VarCopyProc:=@DoVarCopy;
  3869. // Typinfo Variant support
  3870. OnGetVariantProp:=@GetVariantprop;
  3871. OnSetVariantProp:=@SetVariantprop;
  3872. OnSetPropValue:=@SetPropValue;
  3873. OnGetPropValue:=@GetPropValue;
  3874. InvalidCustomVariantType:=TCustomVariantType(-1);
  3875. SetLength(customvarianttypes,CFirstUserType);
  3876. Finalization
  3877. EnterCriticalSection(customvarianttypelock);
  3878. try
  3879. for i:=0 to high(customvarianttypes) do
  3880. if customvarianttypes[i]<>InvalidCustomVariantType then
  3881. customvarianttypes[i].Free;
  3882. finally
  3883. LeaveCriticalSection(customvarianttypelock);
  3884. end;
  3885. UnSetSysVariantManager;
  3886. DoneCriticalSection(customvarianttypelock);
  3887. end.