variants.pp 128 KB

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