variants.pp 122 KB

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