variants.pp 123 KB

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