com_variant.c 27 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Author: Wez Furlong <wez@thebrainroom.com> |
  16. +----------------------------------------------------------------------+
  17. */
  18. #ifdef HAVE_CONFIG_H
  19. #include "config.h"
  20. #endif
  21. #include "php.h"
  22. #include "php_ini.h"
  23. #include "ext/standard/info.h"
  24. #include "php_com_dotnet.h"
  25. #include "php_com_dotnet_internal.h"
  26. /* create an automation SafeArray from a PHP array.
  27. * Only creates a single-dimensional array of variants.
  28. * The keys of the PHP hash MUST be numeric. If the array
  29. * is sparse, then the gaps will be filled with NULL variants */
  30. static void safe_array_from_zval(VARIANT *v, zval *z, int codepage)
  31. {
  32. SAFEARRAY *sa = NULL;
  33. SAFEARRAYBOUND bound;
  34. HashPosition pos;
  35. int keytype;
  36. zend_string *strindex;
  37. zend_ulong intindex = 0;
  38. VARIANT *va;
  39. zval *item;
  40. /* find the largest array index, and assert that all keys are integers */
  41. zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(z), &pos);
  42. for (;; zend_hash_move_forward_ex(Z_ARRVAL_P(z), &pos)) {
  43. keytype = zend_hash_get_current_key_ex(Z_ARRVAL_P(z), &strindex, &intindex, &pos);
  44. if (HASH_KEY_IS_STRING == keytype) {
  45. goto bogus;
  46. } else if (HASH_KEY_NON_EXISTENT == keytype) {
  47. break;
  48. } else if (intindex > UINT_MAX) {
  49. php_error_docref(NULL, E_WARNING, "COM: max number %u of elements in safe array exceeded", UINT_MAX);
  50. break;
  51. }
  52. }
  53. /* allocate the structure */
  54. bound.lLbound = 0;
  55. bound.cElements = zend_hash_num_elements(Z_ARRVAL_P(z));
  56. sa = SafeArrayCreate(VT_VARIANT, 1, &bound);
  57. /* get a lock on the array itself */
  58. SafeArrayAccessData(sa, &va);
  59. va = (VARIANT*)sa->pvData;
  60. /* now fill it in */
  61. zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(z), &pos);
  62. for (;; zend_hash_move_forward_ex(Z_ARRVAL_P(z), &pos)) {
  63. if (NULL == (item = zend_hash_get_current_data_ex(Z_ARRVAL_P(z), &pos))) {
  64. break;
  65. }
  66. zend_hash_get_current_key_ex(Z_ARRVAL_P(z), &strindex, &intindex, &pos);
  67. php_com_variant_from_zval(&va[intindex], item, codepage);
  68. }
  69. /* Unlock it and stuff it into our variant */
  70. SafeArrayUnaccessData(sa);
  71. V_VT(v) = VT_ARRAY|VT_VARIANT;
  72. V_ARRAY(v) = sa;
  73. return;
  74. bogus:
  75. php_error_docref(NULL, E_WARNING, "COM: converting from PHP array to VARIANT array; only arrays with numeric keys are allowed");
  76. V_VT(v) = VT_NULL;
  77. if (sa) {
  78. SafeArrayUnlock(sa);
  79. SafeArrayDestroy(sa);
  80. }
  81. }
  82. PHP_COM_DOTNET_API void php_com_variant_from_zval(VARIANT *v, zval *z, int codepage)
  83. {
  84. php_com_dotnet_object *obj;
  85. zend_uchar ztype = IS_NULL;
  86. if (z) {
  87. ZVAL_DEREF(z);
  88. ztype = Z_TYPE_P(z);
  89. }
  90. switch (ztype) {
  91. case IS_NULL:
  92. V_VT(v) = VT_NULL;
  93. break;
  94. case IS_FALSE:
  95. V_VT(v) = VT_BOOL;
  96. V_BOOL(v) = VARIANT_FALSE;
  97. break;
  98. case IS_TRUE:
  99. V_VT(v) = VT_BOOL;
  100. V_BOOL(v) = VARIANT_TRUE;
  101. break;
  102. case IS_OBJECT:
  103. if (php_com_is_valid_object(z)) {
  104. obj = CDNO_FETCH(z);
  105. if (V_VT(&obj->v) == VT_DISPATCH) {
  106. /* pass the underlying object */
  107. V_VT(v) = VT_DISPATCH;
  108. if (V_DISPATCH(&obj->v)) {
  109. IDispatch_AddRef(V_DISPATCH(&obj->v));
  110. }
  111. V_DISPATCH(v) = V_DISPATCH(&obj->v);
  112. } else {
  113. /* pass the variant by reference */
  114. V_VT(v) = VT_VARIANT | VT_BYREF;
  115. V_VARIANTREF(v) = &obj->v;
  116. }
  117. } else {
  118. /* export the PHP object using our COM wrapper */
  119. V_VT(v) = VT_DISPATCH;
  120. V_DISPATCH(v) = php_com_wrapper_export(z);
  121. }
  122. break;
  123. case IS_ARRAY:
  124. /* map as safe array */
  125. safe_array_from_zval(v, z, codepage);
  126. break;
  127. case IS_LONG:
  128. #if SIZEOF_ZEND_LONG == 4
  129. V_VT(v) = VT_I4;
  130. V_I4(v) = Z_LVAL_P(z);
  131. #else
  132. V_VT(v) = VT_I8;
  133. V_I8(v) = Z_LVAL_P(z);
  134. #endif
  135. break;
  136. case IS_DOUBLE:
  137. V_VT(v) = VT_R8;
  138. V_R8(v) = Z_DVAL_P(z);
  139. break;
  140. case IS_STRING:
  141. V_VT(v) = VT_BSTR;
  142. V_BSTR(v) = php_com_string_to_bstr(Z_STR_P(z), codepage);
  143. break;
  144. case IS_RESOURCE:
  145. case IS_CONSTANT_AST:
  146. default:
  147. V_VT(v) = VT_NULL;
  148. break;
  149. }
  150. }
  151. PHP_COM_DOTNET_API int php_com_zval_from_variant(zval *z, VARIANT *v, int codepage)
  152. {
  153. OLECHAR *olestring = NULL;
  154. int ret = SUCCESS;
  155. switch (V_VT(v)) {
  156. case VT_EMPTY:
  157. case VT_NULL:
  158. case VT_VOID:
  159. ZVAL_NULL(z);
  160. break;
  161. case VT_UI1:
  162. ZVAL_LONG(z, (zend_long)V_UI1(v));
  163. break;
  164. case VT_I1:
  165. ZVAL_LONG(z, (zend_long)V_I1(v));
  166. break;
  167. case VT_UI2:
  168. ZVAL_LONG(z, (zend_long)V_UI2(v));
  169. break;
  170. case VT_I2:
  171. ZVAL_LONG(z, (zend_long)V_I2(v));
  172. break;
  173. case VT_UI4: /* TODO: promote to double if large? */
  174. ZVAL_LONG(z, (long)V_UI4(v));
  175. break;
  176. case VT_I4:
  177. ZVAL_LONG(z, (long)V_I4(v));
  178. break;
  179. #if SIZEOF_ZEND_LONG == 8
  180. case VT_UI8:
  181. ZVAL_LONG(z, (zend_long)V_UI8(v));
  182. break;
  183. case VT_I8:
  184. ZVAL_LONG(z, (zend_long)V_I8(v));
  185. break;
  186. #endif
  187. case VT_INT:
  188. ZVAL_LONG(z, V_INT(v));
  189. break;
  190. case VT_UINT: /* TODO: promote to double if large? */
  191. ZVAL_LONG(z, (zend_long)V_UINT(v));
  192. break;
  193. case VT_R4:
  194. ZVAL_DOUBLE(z, (double)V_R4(v));
  195. break;
  196. case VT_R8:
  197. ZVAL_DOUBLE(z, V_R8(v));
  198. break;
  199. case VT_BOOL:
  200. ZVAL_BOOL(z, V_BOOL(v) ? 1 : 0);
  201. break;
  202. case VT_BSTR:
  203. olestring = V_BSTR(v);
  204. if (olestring) {
  205. zend_string *str = php_com_bstr_to_string(olestring, codepage);
  206. ZVAL_STR(z, str);
  207. olestring = NULL;
  208. }
  209. break;
  210. case VT_UNKNOWN:
  211. if (V_UNKNOWN(v) != NULL) {
  212. IDispatch *disp;
  213. if (SUCCEEDED(IUnknown_QueryInterface(V_UNKNOWN(v), &IID_IDispatch, &disp))) {
  214. php_com_wrap_dispatch(z, disp, codepage);
  215. IDispatch_Release(disp);
  216. } else {
  217. ret = FAILURE;
  218. }
  219. }
  220. break;
  221. case VT_DISPATCH:
  222. if (V_DISPATCH(v) != NULL) {
  223. php_com_wrap_dispatch(z, V_DISPATCH(v), codepage);
  224. }
  225. break;
  226. case VT_VARIANT:
  227. /* points to another variant */
  228. return php_com_zval_from_variant(z, V_VARIANTREF(v), codepage);
  229. default:
  230. php_com_wrap_variant(z, v, codepage);
  231. }
  232. if (olestring) {
  233. efree(olestring);
  234. }
  235. if (ret == FAILURE) {
  236. php_error_docref(NULL, E_WARNING, "variant->zval: conversion from 0x%x ret=%d", V_VT(v), ret);
  237. }
  238. return ret;
  239. }
  240. PHP_COM_DOTNET_API int php_com_copy_variant(VARIANT *dstvar, VARIANT *srcvar)
  241. {
  242. int ret = SUCCESS;
  243. switch (V_VT(dstvar) & ~VT_BYREF) {
  244. case VT_EMPTY:
  245. case VT_NULL:
  246. case VT_VOID:
  247. /* should not be possible */
  248. break;
  249. case VT_UI1:
  250. if (V_VT(dstvar) & VT_BYREF) {
  251. *V_UI1REF(dstvar) = V_UI1(srcvar);
  252. } else {
  253. V_UI1(dstvar) = V_UI1(srcvar);
  254. }
  255. break;
  256. case VT_I1:
  257. if (V_VT(dstvar) & VT_BYREF) {
  258. *V_I1REF(dstvar) = V_I1(srcvar);
  259. } else {
  260. V_I1(dstvar) = V_I1(srcvar);
  261. }
  262. break;
  263. case VT_UI2:
  264. if (V_VT(dstvar) & VT_BYREF) {
  265. *V_UI2REF(dstvar) = V_UI2(srcvar);
  266. } else {
  267. V_UI2(dstvar) = V_UI2(srcvar);
  268. }
  269. break;
  270. case VT_I2:
  271. if (V_VT(dstvar) & VT_BYREF) {
  272. *V_I2REF(dstvar) = V_I2(srcvar);
  273. } else {
  274. V_I2(dstvar) = V_I2(srcvar);
  275. }
  276. break;
  277. case VT_UI4:
  278. if (V_VT(dstvar) & VT_BYREF) {
  279. *V_UI4REF(dstvar) = V_UI4(srcvar);
  280. } else {
  281. V_UI4(dstvar) = V_UI4(srcvar);
  282. }
  283. break;
  284. case VT_I4:
  285. if (V_VT(dstvar) & VT_BYREF) {
  286. *V_I4REF(dstvar) = V_I4(srcvar);
  287. } else {
  288. V_I4(dstvar) = V_I4(srcvar);
  289. }
  290. break;
  291. #if SIZEOF_ZEND_LONG == 8
  292. case VT_UI8:
  293. if (V_VT(dstvar) & VT_BYREF) {
  294. *V_UI8REF(dstvar) = V_UI8(srcvar);
  295. } else {
  296. V_UI8(dstvar) = V_UI8(srcvar);
  297. }
  298. break;
  299. case VT_I8:
  300. if (V_VT(dstvar) & VT_BYREF) {
  301. *V_I8REF(dstvar) = V_I8(srcvar);
  302. } else {
  303. V_I8(dstvar) = V_I8(srcvar);
  304. }
  305. break;
  306. #endif
  307. case VT_INT:
  308. if (V_VT(dstvar) & VT_BYREF) {
  309. *V_INTREF(dstvar) = V_INT(srcvar);
  310. } else {
  311. V_INT(dstvar) = V_INT(srcvar);
  312. }
  313. break;
  314. case VT_UINT:
  315. if (V_VT(dstvar) & VT_BYREF) {
  316. *V_UINTREF(dstvar) = V_UINT(srcvar);
  317. } else {
  318. V_UINT(dstvar) = V_UINT(srcvar);
  319. }
  320. break;
  321. case VT_R4:
  322. if (V_VT(dstvar) & VT_BYREF) {
  323. *V_R4REF(dstvar) = V_R4(srcvar);
  324. } else {
  325. V_R4(dstvar) = V_R4(srcvar);
  326. }
  327. break;
  328. case VT_R8:
  329. if (V_VT(dstvar) & VT_BYREF) {
  330. *V_R8REF(dstvar) = V_R8(srcvar);
  331. } else {
  332. V_R8(dstvar) = V_R8(srcvar);
  333. }
  334. break;
  335. case VT_BOOL:
  336. if (V_VT(dstvar) & VT_BYREF) {
  337. *V_BOOLREF(dstvar) = V_BOOL(srcvar);
  338. } else {
  339. V_BOOL(dstvar) = V_BOOL(srcvar);
  340. }
  341. break;
  342. case VT_BSTR:
  343. if (V_VT(dstvar) & VT_BYREF) {
  344. *V_BSTRREF(dstvar) = V_BSTR(srcvar);
  345. } else {
  346. V_BSTR(dstvar) = V_BSTR(srcvar);
  347. }
  348. break;
  349. case VT_UNKNOWN:
  350. if (V_VT(dstvar) & VT_BYREF) {
  351. *V_UNKNOWNREF(dstvar) = V_UNKNOWN(srcvar);
  352. } else {
  353. V_UNKNOWN(dstvar) = V_UNKNOWN(srcvar);
  354. }
  355. break;
  356. case VT_DISPATCH:
  357. if (V_VT(dstvar) & VT_BYREF) {
  358. *V_DISPATCHREF(dstvar) = V_DISPATCH(srcvar);
  359. } else {
  360. V_DISPATCH(dstvar) = V_DISPATCH(srcvar);
  361. }
  362. break;
  363. case VT_VARIANT:
  364. return php_com_copy_variant(V_VARIANTREF(dstvar), srcvar);
  365. default:
  366. php_error_docref(NULL, E_WARNING, "variant->variant: failed to copy from 0x%x to 0x%x", V_VT(dstvar), V_VT(srcvar));
  367. ret = FAILURE;
  368. }
  369. return ret;
  370. }
  371. /* {{{ com_variant_create_instance - ctor for new VARIANT() */
  372. PHP_FUNCTION(com_variant_create_instance)
  373. {
  374. /* VARTYPE == unsigned short */ zend_long vt = VT_EMPTY;
  375. zend_long codepage = CP_ACP;
  376. zval *object = getThis();
  377. php_com_dotnet_object *obj;
  378. zval *zvalue = NULL;
  379. HRESULT res;
  380. if (ZEND_NUM_ARGS() == 0) {
  381. /* just leave things as-is - an empty variant */
  382. return;
  383. }
  384. obj = CDNO_FETCH(object);
  385. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(),
  386. "z!|ll", &zvalue, &vt, &codepage)) {
  387. php_com_throw_exception(E_INVALIDARG, "Invalid arguments");
  388. return;
  389. }
  390. php_com_initialize();
  391. if (ZEND_NUM_ARGS() == 3) {
  392. obj->code_page = (int)codepage;
  393. }
  394. if (zvalue) {
  395. php_com_variant_from_zval(&obj->v, zvalue, obj->code_page);
  396. }
  397. /* Only perform conversion if variant not already of type passed */
  398. if ((ZEND_NUM_ARGS() >= 2) && (vt != V_VT(&obj->v))) {
  399. /* If already an array and VT_ARRAY is passed then:
  400. - if only VT_ARRAY passed then do not perform a conversion
  401. - if VT_ARRAY plus other type passed then perform conversion
  402. but will probably fail (original behavior)
  403. */
  404. if ((vt & VT_ARRAY) && (V_VT(&obj->v) & VT_ARRAY)) {
  405. zend_long orig_vt = vt;
  406. vt &= ~VT_ARRAY;
  407. if (vt) {
  408. vt = orig_vt;
  409. }
  410. }
  411. if (vt) {
  412. res = VariantChangeType(&obj->v, &obj->v, 0, (VARTYPE)vt);
  413. if (FAILED(res)) {
  414. char *werr, *msg;
  415. werr = php_win32_error_to_msg(res);
  416. spprintf(&msg, 0, "Variant type conversion failed: %s", werr);
  417. LocalFree(werr);
  418. php_com_throw_exception(res, msg);
  419. efree(msg);
  420. }
  421. }
  422. }
  423. if (V_VT(&obj->v) != VT_DISPATCH && obj->typeinfo) {
  424. ITypeInfo_Release(obj->typeinfo);
  425. obj->typeinfo = NULL;
  426. }
  427. }
  428. /* }}} */
  429. /* {{{ proto void variant_set(object variant, mixed value)
  430. Assigns a new value for a variant object */
  431. PHP_FUNCTION(variant_set)
  432. {
  433. zval *zobj, *zvalue = NULL;
  434. php_com_dotnet_object *obj;
  435. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(),
  436. "Oz!", &zobj, php_com_variant_class_entry, &zvalue)) {
  437. return;
  438. }
  439. obj = CDNO_FETCH(zobj);
  440. /* dtor the old value */
  441. if (obj->typeinfo) {
  442. ITypeInfo_Release(obj->typeinfo);
  443. obj->typeinfo = NULL;
  444. }
  445. if (obj->sink_dispatch) {
  446. php_com_object_enable_event_sink(obj, FALSE);
  447. IDispatch_Release(obj->sink_dispatch);
  448. obj->sink_dispatch = NULL;
  449. }
  450. VariantClear(&obj->v);
  451. php_com_variant_from_zval(&obj->v, zvalue, obj->code_page);
  452. /* remember we modified this variant */
  453. obj->modified = 1;
  454. }
  455. /* }}} */
  456. enum variant_binary_opcode {
  457. VOP_ADD, VOP_CAT, VOP_SUB, VOP_MUL, VOP_AND, VOP_DIV,
  458. VOP_EQV, VOP_IDIV, VOP_IMP, VOP_MOD, VOP_OR, VOP_POW,
  459. VOP_XOR
  460. };
  461. enum variant_unary_opcode {
  462. VOP_ABS, VOP_FIX, VOP_INT, VOP_NEG, VOP_NOT
  463. };
  464. static void variant_binary_operation(enum variant_binary_opcode op, INTERNAL_FUNCTION_PARAMETERS) /* {{{ */
  465. {
  466. VARIANT vres;
  467. VARIANT left_val, right_val;
  468. VARIANT *vleft = NULL, *vright = NULL;
  469. zval *zleft = NULL, *zright = NULL;
  470. php_com_dotnet_object *obj;
  471. HRESULT result;
  472. int codepage = CP_ACP;
  473. VariantInit(&left_val);
  474. VariantInit(&right_val);
  475. VariantInit(&vres);
  476. if (SUCCESS == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
  477. ZEND_NUM_ARGS(), "OO", &zleft, php_com_variant_class_entry,
  478. &zright, php_com_variant_class_entry)) {
  479. obj = CDNO_FETCH(zleft);
  480. vleft = &obj->v;
  481. obj = CDNO_FETCH(zright);
  482. vright = &obj->v;
  483. } else if (SUCCESS == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
  484. ZEND_NUM_ARGS(), "Oz!", &zleft, php_com_variant_class_entry,
  485. &zright)) {
  486. obj = CDNO_FETCH(zleft);
  487. vleft = &obj->v;
  488. vright = &right_val;
  489. php_com_variant_from_zval(vright, zright, codepage);
  490. } else if (SUCCESS == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
  491. ZEND_NUM_ARGS(), "z!O", &zleft, &zright, php_com_variant_class_entry)) {
  492. obj = CDNO_FETCH(zright);
  493. vright = &obj->v;
  494. vleft = &left_val;
  495. php_com_variant_from_zval(vleft, zleft, codepage);
  496. } else if (SUCCESS == zend_parse_parameters(ZEND_NUM_ARGS(),
  497. "z!z!", &zleft, &zright)) {
  498. vleft = &left_val;
  499. php_com_variant_from_zval(vleft, zleft, codepage);
  500. vright = &right_val;
  501. php_com_variant_from_zval(vright, zright, codepage);
  502. } else {
  503. return;
  504. }
  505. switch (op) {
  506. case VOP_ADD:
  507. result = VarAdd(vleft, vright, &vres);
  508. break;
  509. case VOP_CAT:
  510. result = VarCat(vleft, vright, &vres);
  511. break;
  512. case VOP_SUB:
  513. result = VarSub(vleft, vright, &vres);
  514. break;
  515. case VOP_MUL:
  516. result = VarMul(vleft, vright, &vres);
  517. break;
  518. case VOP_AND:
  519. result = VarAnd(vleft, vright, &vres);
  520. break;
  521. case VOP_DIV:
  522. result = VarDiv(vleft, vright, &vres);
  523. break;
  524. case VOP_EQV:
  525. result = VarEqv(vleft, vright, &vres);
  526. break;
  527. case VOP_IDIV:
  528. result = VarIdiv(vleft, vright, &vres);
  529. break;
  530. case VOP_IMP:
  531. result = VarImp(vleft, vright, &vres);
  532. break;
  533. case VOP_MOD:
  534. result = VarMod(vleft, vright, &vres);
  535. break;
  536. case VOP_OR:
  537. result = VarOr(vleft, vright, &vres);
  538. break;
  539. case VOP_POW:
  540. result = VarPow(vleft, vright, &vres);
  541. break;
  542. case VOP_XOR:
  543. result = VarXor(vleft, vright, &vres);
  544. break;
  545. /*Let say it fails as no valid op has been given */
  546. default:
  547. result = E_INVALIDARG;
  548. }
  549. if (SUCCEEDED(result)) {
  550. php_com_wrap_variant(return_value, &vres, codepage);
  551. } else {
  552. php_com_throw_exception(result, NULL);
  553. }
  554. VariantClear(&vres);
  555. VariantClear(&left_val);
  556. VariantClear(&right_val);
  557. }
  558. /* }}} */
  559. /* {{{ proto mixed variant_add(mixed left, mixed right)
  560. "Adds" two variant values together and returns the result */
  561. PHP_FUNCTION(variant_add)
  562. {
  563. variant_binary_operation(VOP_ADD, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  564. }
  565. /* }}} */
  566. /* {{{ proto mixed variant_cat(mixed left, mixed right)
  567. concatenates two variant values together and returns the result */
  568. PHP_FUNCTION(variant_cat)
  569. {
  570. variant_binary_operation(VOP_CAT, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  571. }
  572. /* }}} */
  573. /* {{{ proto mixed variant_sub(mixed left, mixed right)
  574. subtracts the value of the right variant from the left variant value and returns the result */
  575. PHP_FUNCTION(variant_sub)
  576. {
  577. variant_binary_operation(VOP_SUB, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  578. }
  579. /* }}} */
  580. /* {{{ proto mixed variant_mul(mixed left, mixed right)
  581. multiplies the values of the two variants and returns the result */
  582. PHP_FUNCTION(variant_mul)
  583. {
  584. variant_binary_operation(VOP_MUL, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  585. }
  586. /* }}} */
  587. /* {{{ proto mixed variant_and(mixed left, mixed right)
  588. performs a bitwise AND operation between two variants and returns the result */
  589. PHP_FUNCTION(variant_and)
  590. {
  591. variant_binary_operation(VOP_AND, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  592. }
  593. /* }}} */
  594. /* {{{ proto mixed variant_div(mixed left, mixed right)
  595. Returns the result from dividing two variants */
  596. PHP_FUNCTION(variant_div)
  597. {
  598. variant_binary_operation(VOP_DIV, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  599. }
  600. /* }}} */
  601. /* {{{ proto mixed variant_eqv(mixed left, mixed right)
  602. Performs a bitwise equivalence on two variants */
  603. PHP_FUNCTION(variant_eqv)
  604. {
  605. variant_binary_operation(VOP_EQV, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  606. }
  607. /* }}} */
  608. /* {{{ proto mixed variant_idiv(mixed left, mixed right)
  609. Converts variants to integers and then returns the result from dividing them */
  610. PHP_FUNCTION(variant_idiv)
  611. {
  612. variant_binary_operation(VOP_IDIV, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  613. }
  614. /* }}} */
  615. /* {{{ proto mixed variant_imp(mixed left, mixed right)
  616. Performs a bitwise implication on two variants */
  617. PHP_FUNCTION(variant_imp)
  618. {
  619. variant_binary_operation(VOP_IMP, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  620. }
  621. /* }}} */
  622. /* {{{ proto mixed variant_mod(mixed left, mixed right)
  623. Divides two variants and returns only the remainder */
  624. PHP_FUNCTION(variant_mod)
  625. {
  626. variant_binary_operation(VOP_MOD, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  627. }
  628. /* }}} */
  629. /* {{{ proto mixed variant_or(mixed left, mixed right)
  630. Performs a logical disjunction on two variants */
  631. PHP_FUNCTION(variant_or)
  632. {
  633. variant_binary_operation(VOP_OR, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  634. }
  635. /* }}} */
  636. /* {{{ proto mixed variant_pow(mixed left, mixed right)
  637. Returns the result of performing the power function with two variants */
  638. PHP_FUNCTION(variant_pow)
  639. {
  640. variant_binary_operation(VOP_POW, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  641. }
  642. /* }}} */
  643. /* {{{ proto mixed variant_xor(mixed left, mixed right)
  644. Performs a logical exclusion on two variants */
  645. PHP_FUNCTION(variant_xor)
  646. {
  647. variant_binary_operation(VOP_XOR, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  648. }
  649. /* }}} */
  650. static void variant_unary_operation(enum variant_unary_opcode op, INTERNAL_FUNCTION_PARAMETERS) /* {{{ */
  651. {
  652. VARIANT vres;
  653. VARIANT left_val;
  654. VARIANT *vleft = NULL;
  655. zval *zleft = NULL;
  656. php_com_dotnet_object *obj;
  657. HRESULT result;
  658. int codepage = CP_ACP;
  659. VariantInit(&left_val);
  660. VariantInit(&vres);
  661. if (SUCCESS == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
  662. ZEND_NUM_ARGS(), "O", &zleft, php_com_variant_class_entry)) {
  663. obj = CDNO_FETCH(zleft);
  664. vleft = &obj->v;
  665. } else if (SUCCESS == zend_parse_parameters(ZEND_NUM_ARGS(),
  666. "z!", &zleft)) {
  667. vleft = &left_val;
  668. php_com_variant_from_zval(vleft, zleft, codepage);
  669. } else {
  670. return;
  671. }
  672. switch (op) {
  673. case VOP_ABS:
  674. result = VarAbs(vleft, &vres);
  675. break;
  676. case VOP_FIX:
  677. result = VarFix(vleft, &vres);
  678. break;
  679. case VOP_INT:
  680. result = VarInt(vleft, &vres);
  681. break;
  682. case VOP_NEG:
  683. result = VarNeg(vleft, &vres);
  684. break;
  685. case VOP_NOT:
  686. result = VarNot(vleft, &vres);
  687. break;
  688. default:
  689. result = E_INVALIDARG;
  690. }
  691. if (SUCCEEDED(result)) {
  692. php_com_wrap_variant(return_value, &vres, codepage);
  693. } else {
  694. php_com_throw_exception(result, NULL);
  695. }
  696. VariantClear(&vres);
  697. VariantClear(&left_val);
  698. }
  699. /* }}} */
  700. /* {{{ proto mixed variant_abs(mixed left)
  701. Returns the absolute value of a variant */
  702. PHP_FUNCTION(variant_abs)
  703. {
  704. variant_unary_operation(VOP_ABS, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  705. }
  706. /* }}} */
  707. /* {{{ proto mixed variant_fix(mixed left)
  708. Returns the integer part ? of a variant */
  709. PHP_FUNCTION(variant_fix)
  710. {
  711. variant_unary_operation(VOP_FIX, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  712. }
  713. /* }}} */
  714. /* {{{ proto mixed variant_int(mixed left)
  715. Returns the integer portion of a variant */
  716. PHP_FUNCTION(variant_int)
  717. {
  718. variant_unary_operation(VOP_INT, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  719. }
  720. /* }}} */
  721. /* {{{ proto mixed variant_neg(mixed left)
  722. Performs logical negation on a variant */
  723. PHP_FUNCTION(variant_neg)
  724. {
  725. variant_unary_operation(VOP_NEG, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  726. }
  727. /* }}} */
  728. /* {{{ proto mixed variant_not(mixed left)
  729. Performs bitwise not negation on a variant */
  730. PHP_FUNCTION(variant_not)
  731. {
  732. variant_unary_operation(VOP_NOT, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  733. }
  734. /* }}} */
  735. /* {{{ proto mixed variant_round(mixed left, int decimals)
  736. Rounds a variant to the specified number of decimal places */
  737. PHP_FUNCTION(variant_round)
  738. {
  739. VARIANT vres;
  740. VARIANT left_val;
  741. VARIANT *vleft = NULL;
  742. zval *zleft = NULL;
  743. php_com_dotnet_object *obj;
  744. int codepage = CP_ACP;
  745. zend_long decimals = 0;
  746. VariantInit(&left_val);
  747. VariantInit(&vres);
  748. if (SUCCESS == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
  749. ZEND_NUM_ARGS(), "Ol", &zleft, php_com_variant_class_entry, &decimals)) {
  750. obj = CDNO_FETCH(zleft);
  751. vleft = &obj->v;
  752. } else if (SUCCESS == zend_parse_parameters(ZEND_NUM_ARGS(),
  753. "z!l", &zleft, &decimals)) {
  754. vleft = &left_val;
  755. php_com_variant_from_zval(vleft, zleft, codepage);
  756. } else {
  757. return;
  758. }
  759. if (SUCCEEDED(VarRound(vleft, (int)decimals, &vres))) {
  760. php_com_wrap_variant(return_value, &vres, codepage);
  761. }
  762. VariantClear(&vres);
  763. VariantClear(&left_val);
  764. }
  765. /* }}} */
  766. /* {{{ proto int variant_cmp(mixed left, mixed right [, int lcid [, int flags]])
  767. Compares two variants */
  768. PHP_FUNCTION(variant_cmp)
  769. {
  770. VARIANT left_val, right_val;
  771. VARIANT *vleft = NULL, *vright = NULL;
  772. zval *zleft = NULL, *zright = NULL;
  773. php_com_dotnet_object *obj;
  774. int codepage = CP_ACP;
  775. zend_long lcid = LOCALE_SYSTEM_DEFAULT;
  776. zend_long flags = 0;
  777. /* it is safe to ignore the warning for this line; see the comments in com_handlers.c */
  778. STDAPI VarCmp(LPVARIANT pvarLeft, LPVARIANT pvarRight, LCID lcid, DWORD flags);
  779. VariantInit(&left_val);
  780. VariantInit(&right_val);
  781. if (SUCCESS == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
  782. ZEND_NUM_ARGS(), "OO|ll", &zleft, php_com_variant_class_entry,
  783. &zright, php_com_variant_class_entry, &lcid, &flags)) {
  784. obj = CDNO_FETCH(zleft);
  785. vleft = &obj->v;
  786. obj = CDNO_FETCH(zright);
  787. vright = &obj->v;
  788. } else if (SUCCESS == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
  789. ZEND_NUM_ARGS(), "Oz!|ll", &zleft, php_com_variant_class_entry,
  790. &zright, &lcid, &flags)) {
  791. obj = CDNO_FETCH(zleft);
  792. vleft = &obj->v;
  793. vright = &right_val;
  794. php_com_variant_from_zval(vright, zright, codepage);
  795. } else if (SUCCESS == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
  796. ZEND_NUM_ARGS(), "z!O|ll", &zleft, &zright, php_com_variant_class_entry,
  797. &lcid, &flags)) {
  798. obj = CDNO_FETCH(zright);
  799. vright = &obj->v;
  800. vleft = &left_val;
  801. php_com_variant_from_zval(vleft, zleft, codepage);
  802. } else if (SUCCESS == zend_parse_parameters(ZEND_NUM_ARGS(),
  803. "z!z!|ll", &zleft, &zright, &lcid, &flags)) {
  804. vleft = &left_val;
  805. php_com_variant_from_zval(vleft, zleft, codepage);
  806. vright = &right_val;
  807. php_com_variant_from_zval(vright, zright, codepage);
  808. } else {
  809. return;
  810. }
  811. ZVAL_LONG(return_value, VarCmp(vleft, vright, (LCID)lcid, (ULONG)flags));
  812. VariantClear(&left_val);
  813. VariantClear(&right_val);
  814. }
  815. /* }}} */
  816. /* {{{ proto int variant_date_to_timestamp(object variant)
  817. Converts a variant date/time value to unix timestamp */
  818. PHP_FUNCTION(variant_date_to_timestamp)
  819. {
  820. VARIANT vres;
  821. zval *zleft = NULL;
  822. php_com_dotnet_object *obj;
  823. VariantInit(&vres);
  824. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(),
  825. "O", &zleft, php_com_variant_class_entry)) {
  826. return;
  827. }
  828. obj = CDNO_FETCH(zleft);
  829. if (SUCCEEDED(VariantChangeType(&vres, &obj->v, 0, VT_DATE))) {
  830. SYSTEMTIME systime;
  831. struct tm tmv;
  832. VariantTimeToSystemTime(V_DATE(&vres), &systime);
  833. memset(&tmv, 0, sizeof(tmv));
  834. tmv.tm_year = systime.wYear - 1900;
  835. tmv.tm_mon = systime.wMonth - 1;
  836. tmv.tm_mday = systime.wDay;
  837. tmv.tm_hour = systime.wHour;
  838. tmv.tm_min = systime.wMinute;
  839. tmv.tm_sec = systime.wSecond;
  840. tmv.tm_isdst = -1;
  841. tzset();
  842. RETVAL_LONG(mktime(&tmv));
  843. }
  844. VariantClear(&vres);
  845. }
  846. /* }}} */
  847. /* {{{ proto object variant_date_from_timestamp(int timestamp)
  848. Returns a variant date representation of a unix timestamp */
  849. PHP_FUNCTION(variant_date_from_timestamp)
  850. {
  851. zend_long timestamp;
  852. time_t ttstamp;
  853. SYSTEMTIME systime;
  854. struct tm *tmv;
  855. VARIANT res;
  856. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "l",
  857. &timestamp)) {
  858. return;
  859. }
  860. if (timestamp < 0) {
  861. php_error_docref(NULL, E_WARNING, "Timestamp value must be a positive value.");
  862. RETURN_FALSE;
  863. }
  864. VariantInit(&res);
  865. tzset();
  866. ttstamp = timestamp;
  867. tmv = localtime(&ttstamp);
  868. /* Invalid after 23:59:59, December 31, 3000, UTC */
  869. if (!tmv) {
  870. php_error_docref(NULL, E_WARNING, "Invalid timestamp " ZEND_LONG_FMT, timestamp);
  871. RETURN_FALSE;
  872. }
  873. memset(&systime, 0, sizeof(systime));
  874. systime.wDay = tmv->tm_mday;
  875. systime.wHour = tmv->tm_hour;
  876. systime.wMinute = tmv->tm_min;
  877. systime.wMonth = tmv->tm_mon + 1;
  878. systime.wSecond = tmv->tm_sec;
  879. systime.wYear = tmv->tm_year + 1900;
  880. V_VT(&res) = VT_DATE;
  881. SystemTimeToVariantTime(&systime, &V_DATE(&res));
  882. php_com_wrap_variant(return_value, &res, CP_ACP);
  883. VariantClear(&res);
  884. }
  885. /* }}} */
  886. /* {{{ proto int variant_get_type(object variant)
  887. Returns the VT_XXX type code for a variant */
  888. PHP_FUNCTION(variant_get_type)
  889. {
  890. zval *zobj;
  891. php_com_dotnet_object *obj;
  892. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(),
  893. "O", &zobj, php_com_variant_class_entry)) {
  894. return;
  895. }
  896. obj = CDNO_FETCH(zobj);
  897. RETURN_LONG(V_VT(&obj->v));
  898. }
  899. /* }}} */
  900. /* {{{ proto void variant_set_type(object variant, int type)
  901. Convert a variant into another type. Variant is modified "in-place" */
  902. PHP_FUNCTION(variant_set_type)
  903. {
  904. zval *zobj;
  905. php_com_dotnet_object *obj;
  906. /* VARTYPE == unsigned short */ zend_long vt;
  907. HRESULT res;
  908. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(),
  909. "Ol", &zobj, php_com_variant_class_entry, &vt)) {
  910. return;
  911. }
  912. obj = CDNO_FETCH(zobj);
  913. res = VariantChangeType(&obj->v, &obj->v, 0, (VARTYPE)vt);
  914. if (SUCCEEDED(res)) {
  915. if (vt != VT_DISPATCH && obj->typeinfo) {
  916. ITypeInfo_Release(obj->typeinfo);
  917. obj->typeinfo = NULL;
  918. }
  919. } else {
  920. char *werr, *msg;
  921. werr = php_win32_error_to_msg(res);
  922. spprintf(&msg, 0, "Variant type conversion failed: %s", werr);
  923. LocalFree(werr);
  924. php_com_throw_exception(res, msg);
  925. efree(msg);
  926. }
  927. }
  928. /* }}} */
  929. /* {{{ proto object variant_cast(object variant, int type)
  930. Convert a variant into a new variant object of another type */
  931. PHP_FUNCTION(variant_cast)
  932. {
  933. zval *zobj;
  934. php_com_dotnet_object *obj;
  935. /* VARTYPE == unsigned short */ zend_long vt;
  936. VARIANT vres;
  937. HRESULT res;
  938. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(),
  939. "Ol", &zobj, php_com_variant_class_entry, &vt)) {
  940. return;
  941. }
  942. obj = CDNO_FETCH(zobj);
  943. VariantInit(&vres);
  944. res = VariantChangeType(&vres, &obj->v, 0, (VARTYPE)vt);
  945. if (SUCCEEDED(res)) {
  946. php_com_wrap_variant(return_value, &vres, obj->code_page);
  947. } else {
  948. char *werr, *msg;
  949. werr = php_win32_error_to_msg(res);
  950. spprintf(&msg, 0, "Variant type conversion failed: %s", werr);
  951. LocalFree(werr);
  952. php_com_throw_exception(res, msg);
  953. efree(msg);
  954. }
  955. VariantClear(&vres);
  956. }
  957. /* }}} */