variants.pp 131 KB

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