com_iterator.c 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. /*
  2. +----------------------------------------------------------------------+
  3. | Copyright (c) The PHP Group |
  4. +----------------------------------------------------------------------+
  5. | This source file is subject to version 3.01 of the PHP license, |
  6. | that is bundled with this package in the file LICENSE, and is |
  7. | available through the world-wide-web at the following url: |
  8. | https://www.php.net/license/3_01.txt |
  9. | If you did not receive a copy of the PHP license and are unable to |
  10. | obtain it through the world-wide-web, please send a note to |
  11. | license@php.net so we can mail you a copy immediately. |
  12. +----------------------------------------------------------------------+
  13. | Author: Wez Furlong <wez@thebrainroom.com> |
  14. +----------------------------------------------------------------------+
  15. */
  16. #ifdef HAVE_CONFIG_H
  17. #include "config.h"
  18. #endif
  19. #include "php.h"
  20. #include "php_ini.h"
  21. #include "ext/standard/info.h"
  22. #include "php_com_dotnet.h"
  23. #include "php_com_dotnet_internal.h"
  24. #include "Zend/zend_exceptions.h"
  25. struct php_com_iterator {
  26. zend_object_iterator iter;
  27. IEnumVARIANT *ev;
  28. zend_ulong key;
  29. VARIANT v; /* cached element */
  30. int code_page;
  31. VARIANT safe_array;
  32. VARTYPE sa_type;
  33. LONG sa_max;
  34. zval zdata;
  35. };
  36. static void com_iter_dtor(zend_object_iterator *iter)
  37. {
  38. struct php_com_iterator *I = (struct php_com_iterator*)Z_PTR(iter->data);
  39. if (I->ev) {
  40. IEnumVARIANT_Release(I->ev);
  41. }
  42. VariantClear(&I->v);
  43. VariantClear(&I->safe_array);
  44. zval_ptr_dtor(&I->zdata);
  45. }
  46. static int com_iter_valid(zend_object_iterator *iter)
  47. {
  48. struct php_com_iterator *I = (struct php_com_iterator*)Z_PTR(iter->data);
  49. if (Z_TYPE(I->zdata) != IS_UNDEF) {
  50. return SUCCESS;
  51. }
  52. return FAILURE;
  53. }
  54. static zval* com_iter_get_data(zend_object_iterator *iter)
  55. {
  56. struct php_com_iterator *I = (struct php_com_iterator*)Z_PTR(iter->data);
  57. return &I->zdata;
  58. }
  59. static void com_iter_get_key(zend_object_iterator *iter, zval *key)
  60. {
  61. struct php_com_iterator *I = (struct php_com_iterator*)Z_PTR(iter->data);
  62. if (I->key == (zend_ulong)-1) {
  63. ZVAL_NULL(key);
  64. } else {
  65. ZVAL_LONG(key, I->key);
  66. }
  67. }
  68. static void com_iter_move_forwards(zend_object_iterator *iter)
  69. {
  70. struct php_com_iterator *I = (struct php_com_iterator*)Z_PTR(iter->data);
  71. unsigned long n_fetched;
  72. zval ptr;
  73. /* release current cached element */
  74. VariantClear(&I->v);
  75. if (Z_TYPE(I->zdata) != IS_UNDEF) {
  76. zval_ptr_dtor(&I->zdata);
  77. ZVAL_UNDEF(&I->zdata);
  78. }
  79. if (I->ev) {
  80. /* Get the next element */
  81. if (SUCCEEDED(IEnumVARIANT_Next(I->ev, 1, &I->v, &n_fetched)) && n_fetched > 0) {
  82. I->key++;
  83. } else {
  84. /* indicate that there are no more items */
  85. I->key = (zend_ulong)-1;
  86. return;
  87. }
  88. } else {
  89. /* safe array */
  90. if (I->key >= (ULONG) I->sa_max) {
  91. I->key = (zend_ulong)-1;
  92. return;
  93. }
  94. I->key++;
  95. if (php_com_safearray_get_elem(&I->safe_array, &I->v, (LONG)I->key) == 0) {
  96. I->key = (zend_ulong)-1;
  97. return;
  98. }
  99. }
  100. ZVAL_NULL(&ptr);
  101. php_com_zval_from_variant(&ptr, &I->v, I->code_page);
  102. /* php_com_wrap_variant(ptr, &I->v, I->code_page); */
  103. ZVAL_COPY_VALUE(&I->zdata, &ptr);
  104. }
  105. static const zend_object_iterator_funcs com_iter_funcs = {
  106. com_iter_dtor,
  107. com_iter_valid,
  108. com_iter_get_data,
  109. com_iter_get_key,
  110. com_iter_move_forwards,
  111. NULL,
  112. NULL, /* get_gc */
  113. };
  114. zend_object_iterator *php_com_iter_get(zend_class_entry *ce, zval *object, int by_ref)
  115. {
  116. php_com_dotnet_object *obj;
  117. struct php_com_iterator *I;
  118. IEnumVARIANT *iev = NULL;
  119. DISPPARAMS dp;
  120. VARIANT v;
  121. unsigned long n_fetched;
  122. zval ptr;
  123. if (by_ref) {
  124. zend_throw_error(NULL, "An iterator cannot be used with foreach by reference");
  125. return NULL;
  126. }
  127. obj = CDNO_FETCH(object);
  128. if (V_VT(&obj->v) != VT_DISPATCH && !V_ISARRAY(&obj->v)) {
  129. /* TODO Promote to TypeError? */
  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. /* TODO Promote to ValueError? */
  149. php_error_docref(NULL, E_WARNING,
  150. "Can only handle single dimension variant arrays (this array has %d)", dims);
  151. goto fail;
  152. }
  153. /* same semantics as foreach on a PHP array;
  154. * make a copy and enumerate that copy */
  155. VariantCopy(&I->safe_array, &obj->v);
  156. /* determine the key value for the array */
  157. SafeArrayGetLBound(V_ARRAY(&I->safe_array), 1, &bound);
  158. SafeArrayGetUBound(V_ARRAY(&I->safe_array), 1, &I->sa_max);
  159. /* pre-fetch the element */
  160. if (I->sa_max >= bound && php_com_safearray_get_elem(&I->safe_array, &I->v, bound)) {
  161. I->key = bound;
  162. ZVAL_NULL(&ptr);
  163. php_com_zval_from_variant(&ptr, &I->v, I->code_page);
  164. ZVAL_COPY_VALUE(&I->zdata, &ptr);
  165. } else {
  166. I->key = (zend_ulong)-1;
  167. }
  168. } else {
  169. /* can we enumerate it? */
  170. if (FAILED(IDispatch_Invoke(V_DISPATCH(&obj->v), DISPID_NEWENUM,
  171. &IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_METHOD|DISPATCH_PROPERTYGET,
  172. &dp, &v, NULL, NULL))) {
  173. goto fail;
  174. }
  175. /* get something useful out of it */
  176. if (V_VT(&v) == VT_UNKNOWN) {
  177. IUnknown_QueryInterface(V_UNKNOWN(&v), &IID_IEnumVARIANT, (void**)&iev);
  178. } else if (V_VT(&v) == VT_DISPATCH) {
  179. IDispatch_QueryInterface(V_DISPATCH(&v), &IID_IEnumVARIANT, (void**)&iev);
  180. }
  181. VariantClear(&v);
  182. if (iev == NULL) {
  183. goto fail;
  184. }
  185. I->ev = iev;
  186. /* Get the first element now */
  187. if (SUCCEEDED(IEnumVARIANT_Next(I->ev, 1, &I->v, &n_fetched)) && n_fetched > 0) {
  188. /* indicate that we have element 0 */
  189. I->key = 0;
  190. ZVAL_NULL(&ptr);
  191. php_com_zval_from_variant(&ptr, &I->v, I->code_page);
  192. ZVAL_COPY_VALUE(&I->zdata, &ptr);
  193. } else {
  194. /* indicate that there are no more items */
  195. I->key = (zend_ulong)-1;
  196. }
  197. }
  198. return &I->iter;
  199. fail:
  200. if (I) {
  201. VariantClear(&I->safe_array);
  202. VariantClear(&I->v);
  203. efree(I);
  204. }
  205. return NULL;
  206. }