com_iterator.c 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  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. #include "Zend/zend_exceptions.h"
  27. struct php_com_iterator {
  28. zend_object_iterator iter;
  29. IEnumVARIANT *ev;
  30. zend_ulong key;
  31. VARIANT v; /* cached element */
  32. int code_page;
  33. VARIANT safe_array;
  34. VARTYPE sa_type;
  35. LONG sa_max;
  36. zval zdata;
  37. };
  38. static void com_iter_dtor(zend_object_iterator *iter)
  39. {
  40. struct php_com_iterator *I = (struct php_com_iterator*)Z_PTR(iter->data);
  41. if (I->ev) {
  42. IEnumVARIANT_Release(I->ev);
  43. }
  44. VariantClear(&I->v);
  45. VariantClear(&I->safe_array);
  46. zval_ptr_dtor(&I->zdata);
  47. }
  48. static int com_iter_valid(zend_object_iterator *iter)
  49. {
  50. struct php_com_iterator *I = (struct php_com_iterator*)Z_PTR(iter->data);
  51. if (Z_TYPE(I->zdata) != IS_UNDEF) {
  52. return SUCCESS;
  53. }
  54. return FAILURE;
  55. }
  56. static zval* com_iter_get_data(zend_object_iterator *iter)
  57. {
  58. struct php_com_iterator *I = (struct php_com_iterator*)Z_PTR(iter->data);
  59. return &I->zdata;
  60. }
  61. static void com_iter_get_key(zend_object_iterator *iter, zval *key)
  62. {
  63. struct php_com_iterator *I = (struct php_com_iterator*)Z_PTR(iter->data);
  64. if (I->key == (zend_ulong)-1) {
  65. ZVAL_NULL(key);
  66. } else {
  67. ZVAL_LONG(key, I->key);
  68. }
  69. }
  70. static void com_iter_move_forwards(zend_object_iterator *iter)
  71. {
  72. struct php_com_iterator *I = (struct php_com_iterator*)Z_PTR(iter->data);
  73. unsigned long n_fetched;
  74. zval ptr;
  75. /* release current cached element */
  76. VariantClear(&I->v);
  77. if (Z_TYPE(I->zdata) != IS_UNDEF) {
  78. zval_ptr_dtor(&I->zdata);
  79. ZVAL_UNDEF(&I->zdata);
  80. }
  81. if (I->ev) {
  82. /* Get the next element */
  83. if (SUCCEEDED(IEnumVARIANT_Next(I->ev, 1, &I->v, &n_fetched)) && n_fetched > 0) {
  84. I->key++;
  85. } else {
  86. /* indicate that there are no more items */
  87. I->key = (ulong)-1;
  88. return;
  89. }
  90. } else {
  91. /* safe array */
  92. if (I->key >= (ULONG) I->sa_max) {
  93. I->key = (ulong)-1;
  94. return;
  95. }
  96. I->key++;
  97. if (php_com_safearray_get_elem(&I->safe_array, &I->v, (LONG)I->key) == 0) {
  98. I->key = (ulong)-1;
  99. return;
  100. }
  101. }
  102. ZVAL_NULL(&ptr);
  103. php_com_zval_from_variant(&ptr, &I->v, I->code_page);
  104. /* php_com_wrap_variant(ptr, &I->v, I->code_page); */
  105. ZVAL_COPY_VALUE(&I->zdata, &ptr);
  106. }
  107. static const zend_object_iterator_funcs com_iter_funcs = {
  108. com_iter_dtor,
  109. com_iter_valid,
  110. com_iter_get_data,
  111. com_iter_get_key,
  112. com_iter_move_forwards,
  113. NULL
  114. };
  115. zend_object_iterator *php_com_iter_get(zend_class_entry *ce, zval *object, int by_ref)
  116. {
  117. php_com_dotnet_object *obj;
  118. struct php_com_iterator *I;
  119. IEnumVARIANT *iev = NULL;
  120. DISPPARAMS dp;
  121. VARIANT v;
  122. unsigned long n_fetched;
  123. zval ptr;
  124. if (by_ref) {
  125. zend_throw_error(NULL, "An iterator cannot be used with foreach by reference");
  126. return NULL;
  127. }
  128. obj = CDNO_FETCH(object);
  129. if (V_VT(&obj->v) != VT_DISPATCH && !V_ISARRAY(&obj->v)) {
  130. php_error_docref(NULL, E_WARNING, "variant is not an object or array VT=%d", V_VT(&obj->v));
  131. return NULL;
  132. }
  133. memset(&dp, 0, sizeof(dp));
  134. VariantInit(&v);
  135. I = (struct php_com_iterator*)ecalloc(1, sizeof(*I));
  136. zend_iterator_init(&I->iter);
  137. I->iter.funcs = &com_iter_funcs;
  138. Z_PTR(I->iter.data) = I;
  139. I->code_page = obj->code_page;
  140. ZVAL_UNDEF(&I->zdata);
  141. VariantInit(&I->safe_array);
  142. VariantInit(&I->v);
  143. if (V_ISARRAY(&obj->v)) {
  144. LONG bound;
  145. UINT dims;
  146. dims = SafeArrayGetDim(V_ARRAY(&obj->v));
  147. if (dims != 1) {
  148. php_error_docref(NULL, E_WARNING,
  149. "Can only handle single dimension variant arrays (this array has %d)", dims);
  150. goto fail;
  151. }
  152. /* same semantics as foreach on a PHP array;
  153. * make a copy and enumerate that copy */
  154. VariantCopy(&I->safe_array, &obj->v);
  155. /* determine the key value for the array */
  156. SafeArrayGetLBound(V_ARRAY(&I->safe_array), 1, &bound);
  157. SafeArrayGetUBound(V_ARRAY(&I->safe_array), 1, &I->sa_max);
  158. /* pre-fetch the element */
  159. if (I->sa_max >= bound && php_com_safearray_get_elem(&I->safe_array, &I->v, bound)) {
  160. I->key = bound;
  161. ZVAL_NULL(&ptr);
  162. php_com_zval_from_variant(&ptr, &I->v, I->code_page);
  163. ZVAL_COPY_VALUE(&I->zdata, &ptr);
  164. } else {
  165. I->key = (ulong)-1;
  166. }
  167. } else {
  168. /* can we enumerate it? */
  169. if (FAILED(IDispatch_Invoke(V_DISPATCH(&obj->v), DISPID_NEWENUM,
  170. &IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_METHOD|DISPATCH_PROPERTYGET,
  171. &dp, &v, NULL, NULL))) {
  172. goto fail;
  173. }
  174. /* get something useful out of it */
  175. if (V_VT(&v) == VT_UNKNOWN) {
  176. IUnknown_QueryInterface(V_UNKNOWN(&v), &IID_IEnumVARIANT, (void**)&iev);
  177. } else if (V_VT(&v) == VT_DISPATCH) {
  178. IDispatch_QueryInterface(V_DISPATCH(&v), &IID_IEnumVARIANT, (void**)&iev);
  179. }
  180. VariantClear(&v);
  181. if (iev == NULL) {
  182. goto fail;
  183. }
  184. I->ev = iev;
  185. /* Get the first element now */
  186. if (SUCCEEDED(IEnumVARIANT_Next(I->ev, 1, &I->v, &n_fetched)) && n_fetched > 0) {
  187. /* indicate that we have element 0 */
  188. I->key = 0;
  189. ZVAL_NULL(&ptr);
  190. php_com_zval_from_variant(&ptr, &I->v, I->code_page);
  191. ZVAL_COPY_VALUE(&I->zdata, &ptr);
  192. } else {
  193. /* indicate that there are no more items */
  194. I->key = (ulong)-1;
  195. }
  196. }
  197. return &I->iter;
  198. fail:
  199. if (I) {
  200. VariantClear(&I->safe_array);
  201. VariantClear(&I->v);
  202. efree(I);
  203. }
  204. return NULL;
  205. }