variant.inc 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624
  1. {
  2. $Id$
  3. This file is part of the Free Pascal run time library.
  4. Copyright (c) 2001 by the Free Pascal development team
  5. This include file contains the implementation for variants
  6. support in FPC as far as it is part of the system unit
  7. See the file COPYING.FPC, included in this distribution,
  8. for details about the copyright.
  9. This program is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  12. **********************************************************************}
  13. var
  14. variantmanager : tvariantmanager;
  15. procedure invalidvariantop;
  16. begin
  17. HandleErrorFrame(221,get_frame);
  18. end;
  19. procedure vardisperror;
  20. begin
  21. HandleErrorFrame(222,get_frame);
  22. end;
  23. { ---------------------------------------------------------------------
  24. Compiler helper routines.
  25. ---------------------------------------------------------------------}
  26. procedure varclear(var v : tvardata);
  27. begin
  28. if not(v.vtype in [varempty,varerror,varnull]) then
  29. invalidvariantop;
  30. end;
  31. procedure variant_init(var v : variant);[Public,Alias:'FPC_VARIANT_INIT'];
  32. begin
  33. { calling the variant manager here is a problem because the static/global variants
  34. are initialized while the variant manager isn't assigned }
  35. fillchar(v,sizeof(variant),0);
  36. end;
  37. procedure variant_clear(var v : variant);[Public,Alias:'FPC_VARIANT_CLEAR'];
  38. begin
  39. variantmanager.varclear(v);
  40. end;
  41. Procedure fpc_write_text_variant(Len : Longint;var f : Text;const v : variant); iocheck; [Public,Alias:'FPC_WRITE_TEXT_VARIANT']; compilerproc;
  42. Begin
  43. If (InOutRes<>0) then
  44. exit;
  45. case TextRec(f).mode of
  46. fmOutput { fmAppend gets changed to fmOutPut in do_open (JM) }:
  47. if len=-1 then
  48. variantmanager.write0variant(f,v)
  49. else
  50. variantmanager.writevariant(f,v,len);
  51. fmInput: InOutRes:=105
  52. else InOutRes:=103;
  53. end;
  54. End;
  55. function fpc_variant_to_dynarray(const v : variant;typeinfo : pointer) : pointer;compilerproc;
  56. begin
  57. end;
  58. function fpc_dynarray_to_variant(const v : variant;typeinfo : pointer) : pointer;compilerproc;
  59. begin
  60. end;
  61. { ---------------------------------------------------------------------
  62. Overloaded operators.
  63. ---------------------------------------------------------------------}
  64. { Integer }
  65. operator :=(const source : byte) dest : variant;
  66. begin
  67. Variant_Init(Dest);
  68. Variantmanager.varfromInt(Dest,Source,1);
  69. end;
  70. operator :=(const source : shortint) dest : variant;
  71. begin
  72. Variant_Init(Dest);
  73. Variantmanager.varfromInt(Dest,Source,-1);
  74. end;
  75. operator :=(const source : word) dest : variant;
  76. begin
  77. Variant_Init(Dest);
  78. Variantmanager.varfromInt(Dest,Source,2);
  79. end;
  80. operator :=(const source : smallint) dest : variant;
  81. begin
  82. Variant_Init(Dest);
  83. Variantmanager.varfromInt(Dest,Source,-2);
  84. end;
  85. operator :=(const source : dword) dest : variant;
  86. begin
  87. Variant_Init(Dest);
  88. Variantmanager.varfromInt(Dest,Source,4);
  89. end;
  90. operator :=(const source : longint) dest : variant;
  91. begin
  92. // Variant_Init(Dest);
  93. Variantmanager.varfromInt(Dest,Source,-4);
  94. end;
  95. operator :=(const source : qword) dest : variant;
  96. begin
  97. Variant_Init(Dest);
  98. Variantmanager.varfromWord64(Dest,Source);
  99. end;
  100. operator :=(const source : int64) dest : variant;
  101. begin
  102. Variant_Init(Dest);
  103. Variantmanager.varfromInt64(Dest,Source);
  104. end;
  105. { Boolean }
  106. operator :=(const source : boolean) dest : variant;
  107. begin
  108. Variant_Init(Dest);
  109. Variantmanager.varfromBool(Dest,Source);
  110. end;
  111. operator :=(const source : wordbool) dest : variant;
  112. begin
  113. Variant_Init(Dest);
  114. Variantmanager.varfromBool(Dest,Boolean(Source));
  115. end;
  116. operator :=(const source : longbool) dest : variant;
  117. begin
  118. Variant_Init(Dest);
  119. Variantmanager.varfromBool(Dest,Boolean(Source));
  120. end;
  121. { Chars }
  122. operator :=(const source : char) dest : variant;
  123. begin
  124. Variant_Init(Dest);
  125. VariantManager.VarFromPStr(Dest,Source);
  126. end;
  127. operator :=(const source : widechar) dest : variant;
  128. begin
  129. Variant_Init(Dest);
  130. VariantManager.VarFromWStr(Dest,Source);
  131. end;
  132. { Strings }
  133. operator :=(const source : shortstring) dest : variant;
  134. begin
  135. Variant_Init(Dest);
  136. VariantManager.VarFromPStr(Dest,Source);
  137. end;
  138. operator :=(const source : ansistring) dest : variant;
  139. begin
  140. Variant_Init(Dest);
  141. VariantManager.VarFromLStr(Dest,Source);
  142. end;
  143. operator :=(const source : widestring) dest : variant;
  144. begin
  145. Variant_Init(Dest);
  146. VariantManager.VarFromWStr(Dest,Source);
  147. end;
  148. { Floats }
  149. operator :=(const source : single) dest : variant;
  150. begin
  151. Variant_Init(Dest);
  152. VariantManager.VarFromReal(Dest,Source);
  153. end;
  154. operator :=(const source : double) dest : variant;
  155. begin
  156. Variant_Init(Dest);
  157. VariantManager.VarFromReal(Dest,Source);
  158. end;
  159. operator :=(const source : extended) dest : variant;
  160. begin
  161. Variant_Init(Dest);
  162. VariantManager.VarFromReal(Dest,Source);
  163. end;
  164. Operator :=(const source : comp) dest : variant;
  165. begin
  166. Variant_Init(Dest);
  167. VariantManager.VarFromReal(Dest,Source);
  168. end;
  169. { Misc. }
  170. { Fixme!!!
  171. operator :=(const source : currency) dest : variant;
  172. begin
  173. end;
  174. operator :=(const source : tdatetime) dest : variant;
  175. begin
  176. end;
  177. }
  178. {**********************************************************************
  179. from Variant assignments
  180. **********************************************************************}
  181. { Integer }
  182. operator :=(const source : variant) dest : byte;
  183. begin
  184. dest:=variantmanager.vartoint(source);
  185. end;
  186. operator :=(const source : variant) dest : shortint;
  187. begin
  188. dest:=variantmanager.vartoint(source);
  189. end;
  190. operator :=(const source : variant) dest : word;
  191. begin
  192. dest:=variantmanager.vartoint(source);
  193. end;
  194. operator :=(const source : variant) dest : smallint;
  195. begin
  196. dest:=variantmanager.vartoint(source);
  197. end;
  198. operator :=(const source : variant) dest : dword;
  199. begin
  200. dest:=variantmanager.vartoint(source);
  201. end;
  202. operator :=(const source : variant) dest : longint;
  203. begin
  204. dest:=variantmanager.vartoint(source);
  205. end;
  206. operator :=(const source : variant) dest : qword;
  207. begin
  208. dest:=variantmanager.vartoword64(source);
  209. end;
  210. operator :=(const source : variant) dest : int64;
  211. begin
  212. dest:=variantmanager.vartoint64(source);
  213. end;
  214. { Boolean }
  215. operator :=(const source : variant) dest : boolean;
  216. begin
  217. dest:=variantmanager.vartobool(source);
  218. end;
  219. operator :=(const source : variant) dest : wordbool;
  220. begin
  221. dest:=variantmanager.vartobool(source);
  222. end;
  223. operator :=(const source : variant) dest : longbool;
  224. begin
  225. dest:=variantmanager.vartobool(source);
  226. end;
  227. { Chars }
  228. operator :=(const source : variant) dest : char;
  229. Var
  230. S : String;
  231. begin
  232. VariantManager.VarToPStr(S,Source);
  233. If Length(S)>0 then
  234. Dest:=S[1];
  235. end;
  236. operator :=(const source : variant) dest : widechar;
  237. Var
  238. WS : WideString;
  239. begin
  240. VariantManager.VarToWStr(WS,Source);
  241. If Length(WS)>0 then
  242. Dest:=WS[1];
  243. end;
  244. { Strings }
  245. operator :=(const source : variant) dest : shortstring;
  246. begin
  247. VariantManager.VarToPStr(Dest,Source);
  248. end;
  249. operator :=(const source : variant) dest : ansistring;
  250. begin
  251. VariantManager.vartolstr(dest,source);
  252. end;
  253. operator :=(const source : variant) dest : widestring;
  254. begin
  255. variantmanager.vartowstr(dest,source);
  256. end;
  257. { Floats }
  258. operator :=(const source : variant) dest : single;
  259. begin
  260. dest:=variantmanager.vartoreal(source);
  261. end;
  262. operator :=(const source : variant) dest : double;
  263. begin
  264. dest:=variantmanager.vartoreal(source);
  265. end;
  266. operator :=(const source : variant) dest : extended;
  267. begin
  268. dest:=variantmanager.vartoreal(source);
  269. end;
  270. operator :=(const source : variant) dest : comp;
  271. begin
  272. dest:=comp(variantmanager.vartoreal(source));
  273. end;
  274. { Misc. }
  275. { FIXME !!!!!!!
  276. operator :=(const source : variant) dest : currency;
  277. begin
  278. dest:=variantmanager.vartocurr(source);
  279. end;
  280. operator :=(const source : variant) dest : tdatetime;
  281. begin
  282. end;
  283. }
  284. {**********************************************************************
  285. Operators
  286. **********************************************************************}
  287. operator or(const op1,op2 : variant) dest : variant;
  288. begin
  289. dest:=op1;
  290. variantmanager.varop(dest,op2,opor);
  291. end;
  292. operator and(const op1,op2 : variant) dest : variant;
  293. begin
  294. dest:=op1;
  295. variantmanager.varop(dest,op2,opand);
  296. end;
  297. operator xor(const op1,op2 : variant) dest : variant;
  298. begin
  299. dest:=op1;
  300. variantmanager.varop(dest,op2,opxor);
  301. end;
  302. operator not(const op : variant) dest : variant;
  303. begin
  304. dest:=op;
  305. variantmanager.varnot(dest);
  306. end;
  307. operator shl(const op1,op2 : variant) dest : variant;
  308. begin
  309. dest:=op1;
  310. variantmanager.varop(dest,op2,opshiftleft);
  311. end;
  312. operator shr(const op1,op2 : variant) dest : variant;
  313. begin
  314. dest:=op1;
  315. variantmanager.varop(dest,op2,opshiftright);
  316. end;
  317. operator +(const op1,op2 : variant) dest : variant;
  318. begin
  319. dest:=op1;
  320. variantmanager.varop(dest,op2,opadd);
  321. end;
  322. operator -(const op1,op2 : variant) dest : variant;
  323. begin
  324. dest:=op1;
  325. variantmanager.varop(dest,op2,opsubtract);
  326. end;
  327. operator *(const op1,op2 : variant) dest : variant;
  328. begin
  329. dest:=op1;
  330. variantmanager.varop(dest,op2,opmultiply);
  331. end;
  332. operator /(const op1,op2 : variant) dest : variant;
  333. begin
  334. dest:=op1;
  335. variantmanager.varop(dest,op2,opdivide);
  336. end;
  337. operator div(const op1,op2 : variant) dest : variant;
  338. begin
  339. dest:=op1;
  340. variantmanager.varop(dest,op2,opintdivide);
  341. end;
  342. operator mod(const op1,op2 : variant) dest : variant;
  343. begin
  344. dest:=op1;
  345. variantmanager.varop(dest,op2,opmodulus);
  346. end;
  347. operator -(const op : variant) dest : variant;
  348. begin
  349. dest:=op;
  350. variantmanager.varneg(dest);
  351. end;
  352. operator =(const op1,op2 : variant) dest : boolean;
  353. begin
  354. dest:=variantmanager.cmpop(op1,op2,opcmpeq);
  355. end;
  356. operator <(const op1,op2 : variant) dest : boolean;
  357. begin
  358. dest:=variantmanager.cmpop(op1,op2,opcmplt);
  359. end;
  360. operator >(const op1,op2 : variant) dest : boolean;
  361. begin
  362. dest:=variantmanager.cmpop(op1,op2,opcmpgt);
  363. end;
  364. operator >=(const op1,op2 : variant) dest : boolean;
  365. begin
  366. dest:=variantmanager.cmpop(op1,op2,opcmpge);
  367. end;
  368. operator <=(const op1,op2 : variant) dest : boolean;
  369. begin
  370. dest:=variantmanager.cmpop(op1,op2,opcmplt);
  371. end;
  372. {**********************************************************************
  373. Variant manager functions
  374. **********************************************************************}
  375. procedure GetVariantManager(var VarMgr: TVariantManager);
  376. begin
  377. VarMgr:=VariantManager;
  378. end;
  379. procedure SetVariantManager(const VarMgr: TVariantManager);
  380. begin
  381. VariantManager:=VarMgr;
  382. end;
  383. function IsVariantManagerSet: Boolean;
  384. var
  385. i : longint;
  386. begin
  387. I:=0;
  388. Result:=True;
  389. While Result and (I<(sizeof(tvariantmanager) div sizeof(pointer))-1) do
  390. begin
  391. Result:=Pointer(ppointer(@variantmanager+i*sizeof(pointer))^)<>Pointer(@invalidvariantop);
  392. Inc(I);
  393. end;
  394. end;
  395. procedure initvariantmanager;
  396. var
  397. i : longint;
  398. begin
  399. VarDispProc:=@vardisperror;
  400. DispCallByIDProc:=@vardisperror;
  401. tvardata(Unassigned).VType:=varEmpty;
  402. tvardata(Null).VType:=varNull;
  403. for i:=0 to (sizeof(tvariantmanager) div sizeof(pointer))-1 do
  404. ppointer(@variantmanager+i*sizeof(pointer))^:=@invalidvariantop;
  405. pointer(variantmanager.varclear):=@varclear
  406. end;
  407. {
  408. $Log$
  409. Revision 1.14 2003-10-04 23:40:42 florian
  410. * write helper comproc for variants fixed
  411. Revision 1.13 2003/09/03 14:09:37 florian
  412. * arm fixes to the common rtl code
  413. * some generic math code fixed
  414. * ...
  415. Revision 1.12 2002/10/10 19:24:28 florian
  416. + write(ln) support for variants added
  417. Revision 1.11 2002/10/09 20:13:26 florian
  418. * hopefully last fix to get things working :/
  419. Revision 1.10 2002/10/09 19:56:01 florian
  420. * variant assignments don't work yet, commented out
  421. Revision 1.9 2002/10/09 19:08:22 florian
  422. + Variant constants Unassigned and Null added
  423. Revision 1.8 2002/10/07 15:10:45 florian
  424. + variant wrappers for cmp operators added
  425. Revision 1.7 2002/10/07 10:27:45 florian
  426. + more variant wrappers added
  427. Revision 1.6 2002/10/06 22:13:55 florian
  428. * wrappers for xor, or and and operator with variants added
  429. Revision 1.5 2002/09/07 15:07:46 peter
  430. * old logs removed and tabs fixed
  431. }