variants.pp 132 KB

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