common_enum.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  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. | http://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. | Authors: Gustavo Lopes <cataphract@php.net> |
  14. +----------------------------------------------------------------------+
  15. */
  16. #ifdef HAVE_CONFIG_H
  17. #include "config.h"
  18. #endif
  19. #include "../intl_cppshims.h"
  20. // Fix build on Windows/old versions of ICU
  21. #include <stdio.h>
  22. #include "common_enum.h"
  23. extern "C" {
  24. #include <zend_interfaces.h>
  25. #include <zend_exceptions.h>
  26. }
  27. zend_class_entry *IntlIterator_ce_ptr;
  28. zend_object_handlers IntlIterator_handlers;
  29. void zoi_with_current_dtor(zend_object_iterator *iter TSRMLS_DC)
  30. {
  31. zoi_with_current *zoiwc = (zoi_with_current*)iter;
  32. if (zoiwc->wrapping_obj) {
  33. /* we have to copy the pointer because zoiwc->wrapping_obj may be
  34. * changed midway the execution of zval_ptr_dtor() */
  35. zval *zwo = zoiwc->wrapping_obj;
  36. /* object is still here, we can rely on it to call this again and
  37. * destroy this object */
  38. zval_ptr_dtor(&zwo);
  39. } else {
  40. /* Object not here anymore (we've been called by the object free handler)
  41. * Note that the iterator wrapper objects (that also depend on this
  42. * structure) call this function earlier, in the destruction phase, which
  43. * precedes the object free phase. Therefore there's no risk on this
  44. * function being called by the iterator wrapper destructor function and
  45. * not finding the memory of this iterator allocated anymore. */
  46. iter->funcs->invalidate_current(iter TSRMLS_CC);
  47. zoiwc->destroy_it(iter TSRMLS_CC);
  48. efree(iter);
  49. }
  50. }
  51. U_CFUNC int zoi_with_current_valid(zend_object_iterator *iter TSRMLS_DC)
  52. {
  53. return ((zoi_with_current*)iter)->current != NULL ? SUCCESS : FAILURE;
  54. }
  55. U_CFUNC void zoi_with_current_get_current_data(zend_object_iterator *iter, zval ***data TSRMLS_DC)
  56. {
  57. *data = &((zoi_with_current*)iter)->current;
  58. }
  59. U_CFUNC void zoi_with_current_invalidate_current(zend_object_iterator *iter TSRMLS_DC)
  60. {
  61. zoi_with_current *zoi_iter = (zoi_with_current*)iter;
  62. if (zoi_iter->current) {
  63. zval_ptr_dtor(&zoi_iter->current);
  64. zoi_iter->current = NULL; //valid would return FAILURE now
  65. }
  66. }
  67. static void string_enum_current_move_forward(zend_object_iterator *iter TSRMLS_DC)
  68. {
  69. zoi_with_current *zoi_iter = (zoi_with_current*)iter;
  70. INTLITERATOR_METHOD_INIT_VARS;
  71. iter->funcs->invalidate_current(iter TSRMLS_CC);
  72. object = zoi_iter->wrapping_obj;
  73. INTLITERATOR_METHOD_FETCH_OBJECT_NO_CHECK;
  74. int32_t result_length;
  75. const char *result = ((StringEnumeration*)iter->data)->next(
  76. &result_length, INTLITERATOR_ERROR_CODE(ii));
  77. intl_error_set_code(NULL, INTLITERATOR_ERROR_CODE(ii) TSRMLS_CC);
  78. if (U_FAILURE(INTLITERATOR_ERROR_CODE(ii))) {
  79. intl_errors_set_custom_msg(INTL_DATA_ERROR_P(ii),
  80. "Error fetching next iteration element", 0 TSRMLS_CC);
  81. } else if (result) {
  82. MAKE_STD_ZVAL(zoi_iter->current);
  83. ZVAL_STRINGL(zoi_iter->current, result, result_length, 1);
  84. } //else we've reached the end of the enum, nothing more is required
  85. }
  86. static void string_enum_rewind(zend_object_iterator *iter TSRMLS_DC)
  87. {
  88. zoi_with_current *zoi_iter = (zoi_with_current*)iter;
  89. INTLITERATOR_METHOD_INIT_VARS;
  90. if (zoi_iter->current) {
  91. iter->funcs->invalidate_current(iter TSRMLS_CC);
  92. }
  93. object = zoi_iter->wrapping_obj;
  94. INTLITERATOR_METHOD_FETCH_OBJECT_NO_CHECK;
  95. ((StringEnumeration*)iter->data)->reset(INTLITERATOR_ERROR_CODE(ii));
  96. intl_error_set_code(NULL, INTLITERATOR_ERROR_CODE(ii) TSRMLS_CC);
  97. if (U_FAILURE(INTLITERATOR_ERROR_CODE(ii))) {
  98. intl_errors_set_custom_msg(INTL_DATA_ERROR_P(ii),
  99. "Error resetting enumeration", 0 TSRMLS_CC);
  100. } else {
  101. iter->funcs->move_forward(iter TSRMLS_CC);
  102. }
  103. }
  104. static void string_enum_destroy_it(zend_object_iterator *iter TSRMLS_DC)
  105. {
  106. delete (StringEnumeration*)iter->data;
  107. }
  108. static zend_object_iterator_funcs string_enum_object_iterator_funcs = {
  109. zoi_with_current_dtor,
  110. zoi_with_current_valid,
  111. zoi_with_current_get_current_data,
  112. NULL,
  113. string_enum_current_move_forward,
  114. string_enum_rewind,
  115. zoi_with_current_invalidate_current
  116. };
  117. U_CFUNC void IntlIterator_from_StringEnumeration(StringEnumeration *se, zval *object TSRMLS_DC)
  118. {
  119. IntlIterator_object *ii;
  120. object_init_ex(object, IntlIterator_ce_ptr);
  121. ii = (IntlIterator_object*)zend_object_store_get_object(object TSRMLS_CC);
  122. ii->iterator = (zend_object_iterator*)emalloc(sizeof(zoi_with_current));
  123. ii->iterator->data = (void*)se;
  124. ii->iterator->funcs = &string_enum_object_iterator_funcs;
  125. ii->iterator->index = 0;
  126. ((zoi_with_current*)ii->iterator)->destroy_it = string_enum_destroy_it;
  127. ((zoi_with_current*)ii->iterator)->wrapping_obj = object;
  128. ((zoi_with_current*)ii->iterator)->current = NULL;
  129. }
  130. static void IntlIterator_objects_free(zend_object *object TSRMLS_DC)
  131. {
  132. IntlIterator_object *ii = (IntlIterator_object*) object;
  133. if (ii->iterator) {
  134. zval **wrapping_objp = &((zoi_with_current*)ii->iterator)->wrapping_obj;
  135. *wrapping_objp = NULL;
  136. ii->iterator->funcs->dtor(ii->iterator TSRMLS_CC);
  137. }
  138. intl_error_reset(INTLITERATOR_ERROR_P(ii) TSRMLS_CC);
  139. zend_object_std_dtor(&ii->zo TSRMLS_CC);
  140. efree(ii);
  141. }
  142. static zend_object_iterator *IntlIterator_get_iterator(
  143. zend_class_entry *ce, zval *object, int by_ref TSRMLS_DC)
  144. {
  145. if (by_ref) {
  146. zend_throw_exception(NULL,
  147. "Iteration by reference is not supported", 0 TSRMLS_CC);
  148. return NULL;
  149. }
  150. IntlIterator_object *ii = (IntlIterator_object*)
  151. zend_object_store_get_object(object TSRMLS_CC);
  152. if (ii->iterator == NULL) {
  153. zend_throw_exception(NULL,
  154. "The IntlIterator is not properly constructed", 0 TSRMLS_CC);
  155. return NULL;
  156. }
  157. zval_add_ref(&object);
  158. return ii->iterator;
  159. }
  160. static zend_object_value IntlIterator_object_create(zend_class_entry *ce TSRMLS_DC)
  161. {
  162. zend_object_value retval;
  163. IntlIterator_object *intern;
  164. intern = (IntlIterator_object*)ecalloc(1, sizeof(IntlIterator_object));
  165. zend_object_std_init(&intern->zo, ce TSRMLS_CC);
  166. #if PHP_VERSION_ID < 50399
  167. zend_hash_copy(intern->zo.properties, &(ce->default_properties),
  168. (copy_ctor_func_t) zval_add_ref, NULL, sizeof(zval*));
  169. #else
  170. object_properties_init((zend_object*) intern, ce);
  171. #endif
  172. intl_error_init(INTLITERATOR_ERROR_P(intern) TSRMLS_CC);
  173. intern->iterator = NULL;
  174. retval.handle = zend_objects_store_put(
  175. intern,
  176. (zend_objects_store_dtor_t)zend_objects_destroy_object,
  177. (zend_objects_free_object_storage_t)IntlIterator_objects_free,
  178. NULL TSRMLS_CC);
  179. retval.handlers = &IntlIterator_handlers;
  180. return retval;
  181. }
  182. static PHP_METHOD(IntlIterator, current)
  183. {
  184. zval **data;
  185. INTLITERATOR_METHOD_INIT_VARS;
  186. if (zend_parse_parameters_none() == FAILURE) {
  187. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  188. "IntlIterator::current: bad arguments", 0 TSRMLS_CC);
  189. return;
  190. }
  191. INTLITERATOR_METHOD_FETCH_OBJECT;
  192. ii->iterator->funcs->get_current_data(ii->iterator, &data TSRMLS_CC);
  193. if (data && *data) {
  194. RETURN_ZVAL(*data, 1, 0);
  195. }
  196. }
  197. static PHP_METHOD(IntlIterator, key)
  198. {
  199. INTLITERATOR_METHOD_INIT_VARS;
  200. if (zend_parse_parameters_none() == FAILURE) {
  201. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  202. "IntlIterator::key: bad arguments", 0 TSRMLS_CC);
  203. return;
  204. }
  205. INTLITERATOR_METHOD_FETCH_OBJECT;
  206. if (ii->iterator->funcs->get_current_key) {
  207. ii->iterator->funcs->get_current_key(ii->iterator, return_value TSRMLS_CC);
  208. } else {
  209. RETURN_LONG(ii->iterator->index);
  210. }
  211. }
  212. static PHP_METHOD(IntlIterator, next)
  213. {
  214. INTLITERATOR_METHOD_INIT_VARS;
  215. if (zend_parse_parameters_none() == FAILURE) {
  216. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  217. "IntlIterator::next: bad arguments", 0 TSRMLS_CC);
  218. return;
  219. }
  220. INTLITERATOR_METHOD_FETCH_OBJECT;
  221. ii->iterator->funcs->move_forward(ii->iterator TSRMLS_CC);
  222. /* foreach also advances the index after the last iteration,
  223. * so I see no problem in incrementing the index here unconditionally */
  224. ii->iterator->index++;
  225. }
  226. static PHP_METHOD(IntlIterator, rewind)
  227. {
  228. INTLITERATOR_METHOD_INIT_VARS;
  229. if (zend_parse_parameters_none() == FAILURE) {
  230. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  231. "IntlIterator::rewind: bad arguments", 0 TSRMLS_CC);
  232. return;
  233. }
  234. INTLITERATOR_METHOD_FETCH_OBJECT;
  235. if (ii->iterator->funcs->rewind) {
  236. ii->iterator->funcs->rewind(ii->iterator TSRMLS_CC);
  237. } else {
  238. intl_errors_set(INTLITERATOR_ERROR_P(ii), U_UNSUPPORTED_ERROR,
  239. "IntlIterator::rewind: rewind not supported", 0 TSRMLS_CC);
  240. }
  241. }
  242. static PHP_METHOD(IntlIterator, valid)
  243. {
  244. INTLITERATOR_METHOD_INIT_VARS;
  245. if (zend_parse_parameters_none() == FAILURE) {
  246. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  247. "IntlIterator::valid: bad arguments", 0 TSRMLS_CC);
  248. return;
  249. }
  250. INTLITERATOR_METHOD_FETCH_OBJECT;
  251. RETURN_BOOL(ii->iterator->funcs->valid(ii->iterator TSRMLS_CC) == SUCCESS);
  252. }
  253. ZEND_BEGIN_ARG_INFO_EX(ainfo_se_void, 0, 0, 0)
  254. ZEND_END_ARG_INFO()
  255. static zend_function_entry IntlIterator_class_functions[] = {
  256. PHP_ME(IntlIterator, current, ainfo_se_void, ZEND_ACC_PUBLIC)
  257. PHP_ME(IntlIterator, key, ainfo_se_void, ZEND_ACC_PUBLIC)
  258. PHP_ME(IntlIterator, next, ainfo_se_void, ZEND_ACC_PUBLIC)
  259. PHP_ME(IntlIterator, rewind, ainfo_se_void, ZEND_ACC_PUBLIC)
  260. PHP_ME(IntlIterator, valid, ainfo_se_void, ZEND_ACC_PUBLIC)
  261. PHP_FE_END
  262. };
  263. /* {{{ intl_register_IntlIterator_class
  264. * Initialize 'IntlIterator' class
  265. */
  266. U_CFUNC void intl_register_IntlIterator_class(TSRMLS_D)
  267. {
  268. zend_class_entry ce;
  269. /* Create and register 'IntlIterator' class. */
  270. INIT_CLASS_ENTRY(ce, "IntlIterator", IntlIterator_class_functions);
  271. ce.create_object = IntlIterator_object_create;
  272. IntlIterator_ce_ptr = zend_register_internal_class(&ce TSRMLS_CC);
  273. IntlIterator_ce_ptr->get_iterator = IntlIterator_get_iterator;
  274. zend_class_implements(IntlIterator_ce_ptr TSRMLS_CC, 1,
  275. zend_ce_iterator);
  276. memcpy(&IntlIterator_handlers, zend_get_std_object_handlers(),
  277. sizeof IntlIterator_handlers);
  278. IntlIterator_handlers.clone_obj = NULL;
  279. }