com_saproxy.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566
  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. /* This module implements a SafeArray proxy which is used internally
  19. * by the engine when resolving multi-dimensional array accesses on
  20. * SafeArray types.
  21. * In addition, the proxy is now able to handle properties of COM objects
  22. * that smell like PHP arrays.
  23. * */
  24. #ifdef HAVE_CONFIG_H
  25. #include "config.h"
  26. #endif
  27. #include "php.h"
  28. #include "php_ini.h"
  29. #include "ext/standard/info.h"
  30. #include "php_com_dotnet.h"
  31. #include "php_com_dotnet_internal.h"
  32. #include "Zend/zend_exceptions.h"
  33. typedef struct {
  34. zend_object std;
  35. /* the object we a proxying for; we hold a refcount to it */
  36. zval *zobj;
  37. php_com_dotnet_object *obj;
  38. /* how many dimensions we are indirecting to get into this element */
  39. LONG dimensions;
  40. /* this is an array whose size_is(dimensions) */
  41. zval *indices;
  42. } php_com_saproxy;
  43. typedef struct {
  44. zend_object_iterator iter;
  45. zval proxy_obj;
  46. zval data;
  47. php_com_saproxy *proxy;
  48. LONG key;
  49. LONG imin, imax;
  50. LONG *indices;
  51. } php_com_saproxy_iter;
  52. #define SA_FETCH(zv) (php_com_saproxy*)Z_OBJ_P(zv)
  53. static inline void clone_indices(php_com_saproxy *dest, php_com_saproxy *src, int ndims)
  54. {
  55. int i;
  56. for (i = 0; i < ndims; i++) {
  57. ZVAL_DUP(&dest->indices[i], &src->indices[i]);
  58. }
  59. }
  60. static zval *saproxy_property_read(zval *object, zval *member, int type, void **cahce_slot, zval *rv)
  61. {
  62. ZVAL_NULL(rv);
  63. php_com_throw_exception(E_INVALIDARG, "safearray has no properties");
  64. return rv;
  65. }
  66. static void saproxy_property_write(zval *object, zval *member, zval *value, void **cache_slot)
  67. {
  68. php_com_throw_exception(E_INVALIDARG, "safearray has no properties");
  69. }
  70. static zval *saproxy_read_dimension(zval *object, zval *offset, int type, zval *rv)
  71. {
  72. php_com_saproxy *proxy = SA_FETCH(object);
  73. UINT dims, i;
  74. SAFEARRAY *sa;
  75. LONG ubound, lbound;
  76. HRESULT res;
  77. ZVAL_NULL(rv);
  78. if (V_VT(&proxy->obj->v) == VT_DISPATCH) {
  79. VARIANT v;
  80. zval *args;
  81. /* prop-get using first dimension as the property name,
  82. * all subsequent dimensions and the offset as parameters */
  83. args = safe_emalloc(proxy->dimensions + 1, sizeof(zval), 0);
  84. for (i = 1; i < (UINT) proxy->dimensions; i++) {
  85. args[i-1] = proxy->indices[i];
  86. }
  87. ZVAL_COPY_VALUE(&args[i-1], offset);
  88. convert_to_string(&proxy->indices[0]);
  89. VariantInit(&v);
  90. res = php_com_do_invoke(proxy->obj, Z_STRVAL(proxy->indices[0]),
  91. Z_STRLEN(proxy->indices[0]), DISPATCH_METHOD|DISPATCH_PROPERTYGET, &v,
  92. proxy->dimensions, args, 0);
  93. efree(args);
  94. if (res == SUCCESS) {
  95. php_com_zval_from_variant(rv, &v, proxy->obj->code_page);
  96. VariantClear(&v);
  97. } else if (res == DISP_E_BADPARAMCOUNT) {
  98. /* return another proxy */
  99. php_com_saproxy_create(object, rv, offset);
  100. }
  101. return rv;
  102. } else if (!V_ISARRAY(&proxy->obj->v)) {
  103. php_com_throw_exception(E_INVALIDARG, "invalid read from com proxy object");
  104. return rv;
  105. }
  106. /* the SafeArray case */
  107. /* offset/index must be an integer */
  108. convert_to_long(offset);
  109. sa = V_ARRAY(&proxy->obj->v);
  110. dims = SafeArrayGetDim(sa);
  111. if ((UINT) proxy->dimensions >= dims) {
  112. /* too many dimensions */
  113. php_com_throw_exception(E_INVALIDARG, "too many dimensions!");
  114. return rv;
  115. }
  116. /* bounds check */
  117. SafeArrayGetLBound(sa, proxy->dimensions, &lbound);
  118. SafeArrayGetUBound(sa, proxy->dimensions, &ubound);
  119. if (Z_LVAL_P(offset) < lbound || Z_LVAL_P(offset) > ubound) {
  120. php_com_throw_exception(DISP_E_BADINDEX, "index out of bounds");
  121. return rv;
  122. }
  123. if (dims - 1 == proxy->dimensions) {
  124. LONG *indices;
  125. VARTYPE vt;
  126. VARIANT v;
  127. VariantInit(&v);
  128. /* we can return a real value */
  129. indices = safe_emalloc(dims, sizeof(LONG), 0);
  130. /* copy indices from proxy */
  131. for (i = 0; i < dims; i++) {
  132. convert_to_long(&proxy->indices[i]);
  133. indices[i] = (LONG)Z_LVAL(proxy->indices[i]);
  134. }
  135. /* add user-supplied index */
  136. indices[dims-1] = (LONG)Z_LVAL_P(offset);
  137. /* now fetch the value */
  138. if (FAILED(SafeArrayGetVartype(sa, &vt)) || vt == VT_EMPTY) {
  139. vt = V_VT(&proxy->obj->v) & ~VT_ARRAY;
  140. }
  141. if (vt == VT_VARIANT) {
  142. res = SafeArrayGetElement(sa, indices, &v);
  143. } else {
  144. V_VT(&v) = vt;
  145. res = SafeArrayGetElement(sa, indices, &v.lVal);
  146. }
  147. efree(indices);
  148. if (SUCCEEDED(res)) {
  149. php_com_wrap_variant(rv, &v, proxy->obj->code_page);
  150. } else {
  151. php_com_throw_exception(res, NULL);
  152. }
  153. VariantClear(&v);
  154. } else {
  155. /* return another proxy */
  156. php_com_saproxy_create(object, rv, offset);
  157. }
  158. return rv;
  159. }
  160. static void saproxy_write_dimension(zval *object, zval *offset, zval *value)
  161. {
  162. php_com_saproxy *proxy = SA_FETCH(object);
  163. UINT dims, i;
  164. HRESULT res;
  165. VARIANT v;
  166. if (V_VT(&proxy->obj->v) == VT_DISPATCH) {
  167. /* We do a prop-set using the first dimension as the property name,
  168. * all subsequent dimensions and offset as parameters, with value as
  169. * the final value */
  170. zval *args = safe_emalloc(proxy->dimensions + 2, sizeof(zval), 0);
  171. for (i = 1; i < (UINT) proxy->dimensions; i++) {
  172. ZVAL_COPY_VALUE(&args[i-1], &proxy->indices[i]);
  173. }
  174. ZVAL_COPY_VALUE(&args[i-1], offset);
  175. ZVAL_COPY_VALUE(&args[i], value);
  176. convert_to_string(&proxy->indices[0]);
  177. VariantInit(&v);
  178. if (SUCCESS == php_com_do_invoke(proxy->obj, Z_STRVAL(proxy->indices[0]),
  179. Z_STRLEN(proxy->indices[0]), DISPATCH_PROPERTYPUT, &v, proxy->dimensions + 1,
  180. args, 0)) {
  181. VariantClear(&v);
  182. }
  183. efree(args);
  184. } else if (V_ISARRAY(&proxy->obj->v)) {
  185. LONG *indices;
  186. VARTYPE vt;
  187. dims = SafeArrayGetDim(V_ARRAY(&proxy->obj->v));
  188. indices = safe_emalloc(dims, sizeof(LONG), 0);
  189. /* copy indices from proxy */
  190. for (i = 0; i < dims; i++) {
  191. convert_to_long(&proxy->indices[i]);
  192. indices[i] = (LONG)Z_LVAL(proxy->indices[i]);
  193. }
  194. /* add user-supplied index */
  195. convert_to_long(offset);
  196. indices[dims-1] = (LONG)Z_LVAL_P(offset);
  197. if (FAILED(SafeArrayGetVartype(V_ARRAY(&proxy->obj->v), &vt)) || vt == VT_EMPTY) {
  198. vt = V_VT(&proxy->obj->v) & ~VT_ARRAY;
  199. }
  200. VariantInit(&v);
  201. php_com_variant_from_zval(&v, value, proxy->obj->code_page);
  202. if (V_VT(&v) != vt) {
  203. VariantChangeType(&v, &v, 0, vt);
  204. }
  205. if (vt == VT_VARIANT) {
  206. res = SafeArrayPutElement(V_ARRAY(&proxy->obj->v), indices, &v);
  207. } else {
  208. res = SafeArrayPutElement(V_ARRAY(&proxy->obj->v), indices, &v.lVal);
  209. }
  210. efree(indices);
  211. VariantClear(&v);
  212. if (FAILED(res)) {
  213. php_com_throw_exception(res, NULL);
  214. }
  215. } else {
  216. php_com_throw_exception(E_NOTIMPL, "invalid write to com proxy object");
  217. }
  218. }
  219. #if 0
  220. static void saproxy_object_set(zval **property, zval *value)
  221. {
  222. }
  223. static zval *saproxy_object_get(zval *property)
  224. {
  225. /* Not yet implemented in the engine */
  226. return NULL;
  227. }
  228. #endif
  229. static int saproxy_property_exists(zval *object, zval *member, int check_empty, void **cache_slot)
  230. {
  231. /* no properties */
  232. return 0;
  233. }
  234. static int saproxy_dimension_exists(zval *object, zval *member, int check_empty)
  235. {
  236. php_error_docref(NULL, E_WARNING, "Operation not yet supported on a COM object");
  237. return 0;
  238. }
  239. static void saproxy_property_delete(zval *object, zval *member, void **cache_slot)
  240. {
  241. php_error_docref(NULL, E_WARNING, "Cannot delete properties from a COM object");
  242. }
  243. static void saproxy_dimension_delete(zval *object, zval *offset)
  244. {
  245. php_error_docref(NULL, E_WARNING, "Cannot delete properties from a COM object");
  246. }
  247. static HashTable *saproxy_properties_get(zval *object)
  248. {
  249. /* no properties */
  250. return NULL;
  251. }
  252. static union _zend_function *saproxy_method_get(zend_object **object, zend_string *name, const zval *key)
  253. {
  254. /* no methods */
  255. return NULL;
  256. }
  257. static int saproxy_call_method(zend_string *method, zend_object *object, INTERNAL_FUNCTION_PARAMETERS)
  258. {
  259. return FAILURE;
  260. }
  261. static union _zend_function *saproxy_constructor_get(zend_object *object)
  262. {
  263. /* user cannot instantiate */
  264. return NULL;
  265. }
  266. static zend_string* saproxy_class_name_get(const zend_object *object)
  267. {
  268. return zend_string_copy(php_com_saproxy_class_entry->name);
  269. }
  270. static int saproxy_objects_compare(zval *object1, zval *object2)
  271. {
  272. return -1;
  273. }
  274. static int saproxy_object_cast(zval *readobj, zval *writeobj, int type)
  275. {
  276. return FAILURE;
  277. }
  278. static int saproxy_count_elements(zval *object, zend_long *count)
  279. {
  280. php_com_saproxy *proxy = SA_FETCH(object);
  281. LONG ubound, lbound;
  282. if (!V_ISARRAY(&proxy->obj->v)) {
  283. return FAILURE;
  284. }
  285. SafeArrayGetLBound(V_ARRAY(&proxy->obj->v), proxy->dimensions, &lbound);
  286. SafeArrayGetUBound(V_ARRAY(&proxy->obj->v), proxy->dimensions, &ubound);
  287. *count = ubound - lbound + 1;
  288. return SUCCESS;
  289. }
  290. static void saproxy_free_storage(zend_object *object)
  291. {
  292. php_com_saproxy *proxy = (php_com_saproxy *)object;
  293. //??? int i;
  294. //???
  295. //??? for (i = 0; i < proxy->dimensions; i++) {
  296. //??? if (proxy->indices) {
  297. //??? FREE_ZVAL(proxy->indices[i]);
  298. //??? }
  299. //??? }
  300. zval_ptr_dtor(proxy->zobj);
  301. efree(proxy->indices);
  302. }
  303. static zend_object* saproxy_clone(zval *object)
  304. {
  305. php_com_saproxy *proxy = (php_com_saproxy *)Z_OBJ_P(object);
  306. php_com_saproxy *cloneproxy;
  307. cloneproxy = emalloc(sizeof(*cloneproxy));
  308. memcpy(cloneproxy, proxy, sizeof(*cloneproxy));
  309. Z_ADDREF_P(cloneproxy->zobj);
  310. cloneproxy->indices = safe_emalloc(cloneproxy->dimensions, sizeof(zval), 0);
  311. clone_indices(cloneproxy, proxy, proxy->dimensions);
  312. return &cloneproxy->std;
  313. }
  314. zend_object_handlers php_com_saproxy_handlers = {
  315. 0,
  316. saproxy_free_storage,
  317. zend_objects_destroy_object,
  318. saproxy_clone,
  319. saproxy_property_read,
  320. saproxy_property_write,
  321. saproxy_read_dimension,
  322. saproxy_write_dimension,
  323. NULL,
  324. NULL, /* saproxy_object_get, */
  325. NULL, /* saproxy_object_set, */
  326. saproxy_property_exists,
  327. saproxy_property_delete,
  328. saproxy_dimension_exists,
  329. saproxy_dimension_delete,
  330. saproxy_properties_get,
  331. saproxy_method_get,
  332. saproxy_call_method,
  333. saproxy_constructor_get,
  334. saproxy_class_name_get,
  335. saproxy_objects_compare,
  336. saproxy_object_cast,
  337. saproxy_count_elements
  338. };
  339. int php_com_saproxy_create(zval *com_object, zval *proxy_out, zval *index)
  340. {
  341. php_com_saproxy *proxy, *rel = NULL;
  342. proxy = ecalloc(1, sizeof(*proxy));
  343. proxy->dimensions = 1;
  344. if (Z_OBJCE_P(com_object) == php_com_saproxy_class_entry) {
  345. rel = SA_FETCH(com_object);
  346. proxy->obj = rel->obj;
  347. proxy->zobj = rel->zobj;
  348. proxy->dimensions += rel->dimensions;
  349. } else {
  350. proxy->obj = CDNO_FETCH(com_object);
  351. proxy->zobj = com_object;
  352. }
  353. Z_ADDREF_P(proxy->zobj);
  354. proxy->indices = safe_emalloc(proxy->dimensions, sizeof(zval), 0);
  355. if (rel) {
  356. clone_indices(proxy, rel, rel->dimensions);
  357. }
  358. ZVAL_DUP(&proxy->indices[proxy->dimensions-1], index);
  359. zend_object_std_init(&proxy->std, php_com_saproxy_class_entry);
  360. proxy->std.handlers = &php_com_saproxy_handlers;
  361. ZVAL_OBJ(proxy_out, &proxy->std);
  362. return 1;
  363. }
  364. /* iterator */
  365. static void saproxy_iter_dtor(zend_object_iterator *iter)
  366. {
  367. php_com_saproxy_iter *I = (php_com_saproxy_iter*)Z_PTR(iter->data);
  368. zval_ptr_dtor(&I->proxy_obj);
  369. efree(I->indices);
  370. efree(I);
  371. }
  372. static int saproxy_iter_valid(zend_object_iterator *iter)
  373. {
  374. php_com_saproxy_iter *I = (php_com_saproxy_iter*)Z_PTR(iter->data);
  375. return (I->key < I->imax) ? SUCCESS : FAILURE;
  376. }
  377. static zval* saproxy_iter_get_data(zend_object_iterator *iter)
  378. {
  379. php_com_saproxy_iter *I = (php_com_saproxy_iter*)Z_PTR(iter->data);
  380. VARIANT v;
  381. VARTYPE vt;
  382. SAFEARRAY *sa;
  383. I->indices[I->proxy->dimensions-1] = I->key;
  384. sa = V_ARRAY(&I->proxy->obj->v);
  385. if (FAILED(SafeArrayGetVartype(sa, &vt)) || vt == VT_EMPTY) {
  386. vt = V_VT(&I->proxy->obj->v) & ~VT_ARRAY;
  387. }
  388. VariantInit(&v);
  389. if (vt == VT_VARIANT) {
  390. SafeArrayGetElement(sa, I->indices, &v);
  391. } else {
  392. V_VT(&v) = vt;
  393. SafeArrayGetElement(sa, I->indices, &v.lVal);
  394. }
  395. ZVAL_NULL(&I->data);
  396. php_com_wrap_variant(&I->data, &v, I->proxy->obj->code_page);
  397. VariantClear(&v);
  398. return &I->data;
  399. }
  400. static void saproxy_iter_get_key(zend_object_iterator *iter, zval *key)
  401. {
  402. php_com_saproxy_iter *I = (php_com_saproxy_iter*)Z_PTR(iter->data);
  403. if (I->key == -1) {
  404. ZVAL_NULL(key);
  405. } else {
  406. ZVAL_LONG(key, I->key);
  407. }
  408. }
  409. static void saproxy_iter_move_forwards(zend_object_iterator *iter)
  410. {
  411. php_com_saproxy_iter *I = (php_com_saproxy_iter*)Z_PTR(iter->data);
  412. if (++I->key >= I->imax) {
  413. I->key = -1;
  414. }
  415. }
  416. static const zend_object_iterator_funcs saproxy_iter_funcs = {
  417. saproxy_iter_dtor,
  418. saproxy_iter_valid,
  419. saproxy_iter_get_data,
  420. saproxy_iter_get_key,
  421. saproxy_iter_move_forwards,
  422. NULL
  423. };
  424. zend_object_iterator *php_com_saproxy_iter_get(zend_class_entry *ce, zval *object, int by_ref)
  425. {
  426. php_com_saproxy *proxy = SA_FETCH(object);
  427. php_com_saproxy_iter *I;
  428. int i;
  429. if (by_ref) {
  430. zend_throw_error(NULL, "An iterator cannot be used with foreach by reference");
  431. return NULL;
  432. }
  433. I = ecalloc(1, sizeof(*I));
  434. I->iter.funcs = &saproxy_iter_funcs;
  435. Z_PTR(I->iter.data) = I;
  436. I->proxy = proxy;
  437. ZVAL_COPY(&I->proxy_obj, object);
  438. I->indices = safe_emalloc(proxy->dimensions + 1, sizeof(LONG), 0);
  439. for (i = 0; i < proxy->dimensions; i++) {
  440. convert_to_long(&proxy->indices[i]);
  441. I->indices[i] = (LONG)Z_LVAL(proxy->indices[i]);
  442. }
  443. SafeArrayGetLBound(V_ARRAY(&proxy->obj->v), proxy->dimensions, &I->imin);
  444. SafeArrayGetUBound(V_ARRAY(&proxy->obj->v), proxy->dimensions, &I->imax);
  445. I->key = I->imin;
  446. return &I->iter;
  447. }