Coverage Report

Created: 2026-04-01 06:49

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/php-src/ext/spl/spl_iterators.c
Line
Count
Source
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
   | Authors: Marcus Boerger <helly@php.net>                              |
14
   +----------------------------------------------------------------------+
15
 */
16
17
#ifdef HAVE_CONFIG_H
18
# include "config.h"
19
#endif
20
21
#include "php.h"
22
#include "zend_exceptions.h"
23
#include "zend_interfaces.h"
24
#include "ext/pcre/php_pcre.h"
25
26
#include "spl_iterators.h"
27
#include "spl_iterators_arginfo.h"
28
#include "spl_array.h" /* For spl_ce_ArrayIterator */
29
#include "spl_exceptions.h"
30
#include "zend_smart_str.h"
31
32
#ifdef accept
33
#undef accept
34
#endif
35
36
PHPAPI zend_class_entry *spl_ce_RecursiveIterator;
37
PHPAPI zend_class_entry *spl_ce_RecursiveIteratorIterator;
38
PHPAPI zend_class_entry *spl_ce_FilterIterator;
39
PHPAPI zend_class_entry *spl_ce_CallbackFilterIterator;
40
PHPAPI zend_class_entry *spl_ce_RecursiveFilterIterator;
41
PHPAPI zend_class_entry *spl_ce_RecursiveCallbackFilterIterator;
42
PHPAPI zend_class_entry *spl_ce_ParentIterator;
43
PHPAPI zend_class_entry *spl_ce_SeekableIterator;
44
PHPAPI zend_class_entry *spl_ce_LimitIterator;
45
PHPAPI zend_class_entry *spl_ce_CachingIterator;
46
PHPAPI zend_class_entry *spl_ce_RecursiveCachingIterator;
47
PHPAPI zend_class_entry *spl_ce_OuterIterator;
48
PHPAPI zend_class_entry *spl_ce_IteratorIterator;
49
PHPAPI zend_class_entry *spl_ce_NoRewindIterator;
50
PHPAPI zend_class_entry *spl_ce_InfiniteIterator;
51
PHPAPI zend_class_entry *spl_ce_EmptyIterator;
52
PHPAPI zend_class_entry *spl_ce_AppendIterator;
53
PHPAPI zend_class_entry *spl_ce_RegexIterator;
54
PHPAPI zend_class_entry *spl_ce_RecursiveRegexIterator;
55
PHPAPI zend_class_entry *spl_ce_RecursiveTreeIterator;
56
57
typedef enum {
58
  RS_NEXT  = 0,
59
  RS_TEST  = 1,
60
  RS_SELF  = 2,
61
  RS_CHILD = 3,
62
  RS_START = 4
63
} RecursiveIteratorState;
64
65
typedef struct _spl_sub_iterator {
66
  zend_object_iterator    *iterator;
67
  zval                    zobject;
68
  zend_class_entry        *ce;
69
  RecursiveIteratorState  state;
70
  zend_function           *haschildren;
71
  zend_function           *getchildren;
72
} spl_sub_iterator;
73
74
typedef struct _spl_recursive_it_object {
75
  spl_sub_iterator         *iterators;
76
  int                      level;
77
  RecursiveIteratorMode    mode;
78
  int                      flags;
79
  int                      max_depth;
80
  bool                in_iteration;
81
  zend_function            *beginIteration;
82
  zend_function            *endIteration;
83
  zend_function            *callHasChildren;
84
  zend_function            *callGetChildren;
85
  zend_function            *beginChildren;
86
  zend_function            *endChildren;
87
  zend_function            *nextElement;
88
  zend_class_entry         *ce;
89
  zend_string              *prefix[6];
90
  zend_string              *postfix[1];
91
  zend_object              std;
92
} spl_recursive_it_object;
93
94
typedef struct _spl_recursive_it_iterator {
95
  zend_object_iterator   intern;
96
} spl_recursive_it_iterator;
97
98
typedef struct _spl_dual_it_object {
99
  struct {
100
    zval                 zobject;
101
    zend_class_entry     *ce;
102
    zend_object          *object;
103
    zend_object_iterator *iterator;
104
  } inner;
105
  struct {
106
    zval                 data;
107
    zval                 key;
108
    zend_long            pos;
109
  } current;
110
  dual_it_type             dit_type;
111
  union {
112
    struct {
113
      zend_long             offset;
114
      zend_long             count;
115
    } limit;
116
    struct {
117
      zend_long             flags; /* CIT_* */
118
      zend_string          *zstr;
119
      zval             zchildren;
120
      zval             zcache;
121
    } caching;
122
    struct {
123
      zval                  zarrayit;
124
      zend_object_iterator *iterator;
125
    } append;
126
    struct {
127
      zend_long        flags;
128
      zend_long        preg_flags;
129
      pcre_cache_entry *pce;
130
      zend_string      *regex;
131
      regex_mode       mode;
132
    } regex;
133
    zend_fcall_info_cache callback_filter;
134
  } u;
135
  zend_object              std;
136
} spl_dual_it_object;
137
138
static zend_object_handlers spl_handlers_rec_it_it;
139
static zend_object_handlers spl_handlers_dual_it;
140
141
1.39k
static inline spl_recursive_it_object *spl_recursive_it_from_obj(zend_object *obj) /* {{{ */ {
142
1.39k
  return (spl_recursive_it_object*)((char*)(obj) - XtOffsetOf(spl_recursive_it_object, std));
143
1.39k
}
144
/* }}} */
145
146
123
#define Z_SPLRECURSIVE_IT_P(zv)  spl_recursive_it_from_obj(Z_OBJ_P((zv)))
147
148
2.33k
static inline spl_dual_it_object *spl_dual_it_from_obj(zend_object *obj) /* {{{ */ {
149
2.33k
  return (spl_dual_it_object*)((char*)(obj) - XtOffsetOf(spl_dual_it_object, std));
150
2.33k
} /* }}} */
151
152
779
#define Z_SPLDUAL_IT_P(zv)  spl_dual_it_from_obj(Z_OBJ_P((zv)))
153
154
#define SPL_FETCH_AND_CHECK_DUAL_IT(var, objzval)                         \
155
620
  do {                                             \
156
620
    spl_dual_it_object *it = Z_SPLDUAL_IT_P(objzval);                    \
157
620
    if (it->dit_type == DIT_Unknown) {                             \
158
0
      zend_throw_error(NULL, "The object is in an invalid state as the parent constructor was not called");   \
159
0
      RETURN_THROWS();                                       \
160
0
    }                                            \
161
620
    (var) = it;                                       \
162
620
  } while (0)
163
164
#define SPL_FETCH_SUB_ELEMENT(var, object, element) \
165
45
  do { \
166
45
    if(!(object)->iterators) { \
167
0
      zend_throw_error(NULL, "The object is in an invalid state as the parent constructor was not called"); \
168
0
      return; \
169
0
    } \
170
45
    (var) = (object)->iterators[(object)->level].element; \
171
45
  } while (0)
172
173
#define SPL_FETCH_SUB_ELEMENT_ADDR(var, object, element) \
174
0
  do { \
175
0
    if(!(object)->iterators) { \
176
0
      zend_throw_error(NULL, "The object is in an invalid state as the parent constructor was not called"); \
177
0
      RETURN_THROWS(); \
178
0
    } \
179
0
    (var) = &(object)->iterators[(object)->level].element; \
180
0
  } while (0)
181
182
45
#define SPL_FETCH_SUB_ITERATOR(var, object) SPL_FETCH_SUB_ELEMENT(var, object, iterator)
183
184
185
static void spl_recursive_it_dtor(zend_object_iterator *_iter)
186
12
{
187
12
  spl_recursive_it_iterator *iter   = (spl_recursive_it_iterator*)_iter;
188
12
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(&iter->intern.data);
189
12
  zend_object_iterator      *sub_iter;
190
191
12
  if (object->iterators) {
192
12
    while (object->level > 0) {
193
0
      if (!Z_ISUNDEF(object->iterators[object->level].zobject)) {
194
0
        sub_iter = object->iterators[object->level].iterator;
195
0
        zend_iterator_dtor(sub_iter);
196
0
        zval_ptr_dtor(&object->iterators[object->level].zobject);
197
0
      }
198
0
      object->level--;
199
0
    }
200
12
    object->iterators = erealloc(object->iterators, sizeof(spl_sub_iterator));
201
12
    object->level = 0;
202
12
  }
203
204
12
  zval_ptr_dtor(&iter->intern.data);
205
12
}
206
207
static zend_result spl_recursive_it_valid_ex(spl_recursive_it_object *object, zval *zthis)
208
33
{
209
33
  zend_object_iterator      *sub_iter;
210
33
  int                       level = object->level;
211
212
33
  if(!object->iterators) {
213
0
    return FAILURE;
214
0
  }
215
45
  while (level >=0) {
216
33
    sub_iter = object->iterators[level].iterator;
217
33
    if (sub_iter->funcs->valid(sub_iter) == SUCCESS) {
218
21
      return SUCCESS;
219
21
    }
220
12
    level--;
221
12
  }
222
12
  if (object->endIteration && object->in_iteration) {
223
0
    zend_call_method_with_0_params(Z_OBJ_P(zthis), object->ce, &object->endIteration, "endIteration", NULL);
224
0
  }
225
12
  object->in_iteration = false;
226
12
  return FAILURE;
227
33
}
228
229
static zend_result spl_recursive_it_valid(zend_object_iterator *iter)
230
33
{
231
33
  return spl_recursive_it_valid_ex(Z_SPLRECURSIVE_IT_P(&iter->data), &iter->data);
232
33
}
233
234
static zval *spl_recursive_it_get_current_data(zend_object_iterator *iter)
235
21
{
236
21
  spl_recursive_it_object *object = Z_SPLRECURSIVE_IT_P(&iter->data);
237
21
  zend_object_iterator *sub_iter = object->iterators[object->level].iterator;
238
239
21
  return sub_iter->funcs->get_current_data(sub_iter);
240
21
}
241
242
static void spl_recursive_it_get_current_key(zend_object_iterator *iter, zval *key)
243
0
{
244
0
  spl_recursive_it_object *object = Z_SPLRECURSIVE_IT_P(&iter->data);
245
0
  zend_object_iterator *sub_iter = object->iterators[object->level].iterator;
246
247
0
  if (sub_iter->funcs->get_current_key) {
248
0
    sub_iter->funcs->get_current_key(sub_iter, key);
249
0
  } else {
250
0
    ZVAL_LONG(key, iter->index);
251
0
  }
252
0
}
253
254
static void spl_recursive_it_move_forward_ex(spl_recursive_it_object *object, zval *zthis)
255
33
{
256
33
  zend_object_iterator      *iterator;
257
33
  zend_class_entry          *ce;
258
33
  zval                      retval, child;
259
33
  zend_object_iterator      *sub_iter;
260
261
33
  SPL_FETCH_SUB_ITERATOR(iterator, object);
262
263
42
  while (!EG(exception)) {
264
60
next_step:
265
60
    iterator = object->iterators[object->level].iterator;
266
60
    switch (object->iterators[object->level].state) {
267
21
      case RS_NEXT:
268
21
        iterator->funcs->move_forward(iterator);
269
21
        if (EG(exception)) {
270
0
          if (!(object->flags & RIT_CATCH_GET_CHILD)) {
271
0
            return;
272
0
          } else {
273
0
            zend_clear_exception();
274
0
          }
275
0
        }
276
21
        ZEND_FALLTHROUGH;
277
42
      case RS_START:
278
42
        if (iterator->funcs->valid(iterator) == FAILURE) {
279
21
          break;
280
21
        }
281
21
        object->iterators[object->level].state = RS_TEST;
282
        /* break; */
283
        /* TODO: Check this is correct */
284
21
        ZEND_FALLTHROUGH;
285
21
      case RS_TEST:
286
21
        if (object->callHasChildren) {
287
0
          zend_call_method_with_0_params(Z_OBJ_P(zthis), object->ce, &object->callHasChildren, "callHasChildren", &retval);
288
21
        } else {
289
21
          zend_class_entry *ce = object->iterators[object->level].ce;
290
21
          zend_object *obj = Z_OBJ(object->iterators[object->level].zobject);
291
21
          zend_function **cache = &object->iterators[object->level].haschildren;
292
293
21
          zend_call_method_with_0_params(obj, ce, cache, "haschildren", &retval);
294
21
        }
295
21
        if (EG(exception)) {
296
0
          if (!(object->flags & RIT_CATCH_GET_CHILD)) {
297
0
            object->iterators[object->level].state = RS_NEXT;
298
0
            return;
299
0
          } else {
300
0
            zend_clear_exception();
301
0
          }
302
0
        }
303
21
        if (Z_TYPE(retval) != IS_UNDEF) {
304
21
          bool has_children = zend_is_true(&retval);
305
21
          zval_ptr_dtor(&retval);
306
21
          if (has_children) {
307
9
            if (object->max_depth == -1 || object->max_depth > object->level) {
308
9
              switch (object->mode) {
309
0
              case RIT_LEAVES_ONLY:
310
0
              case RIT_CHILD_FIRST:
311
0
                object->iterators[object->level].state = RS_CHILD;
312
0
                goto next_step;
313
9
              case RIT_SELF_FIRST:
314
9
                object->iterators[object->level].state = RS_SELF;
315
9
                goto next_step;
316
9
              }
317
9
            } else {
318
              /* do not recurse into */
319
0
              if (object->mode == RIT_LEAVES_ONLY) {
320
                /* this is not a leave, so skip it */
321
0
                object->iterators[object->level].state = RS_NEXT;
322
0
                goto next_step;
323
0
              }
324
0
            }
325
9
          }
326
21
        }
327
12
        if (object->nextElement) {
328
0
          zend_call_method_with_0_params(Z_OBJ_P(zthis), object->ce, &object->nextElement, "nextelement", NULL);
329
0
        }
330
12
        object->iterators[object->level].state = RS_NEXT;
331
12
        if (EG(exception)) {
332
0
          if (!(object->flags & RIT_CATCH_GET_CHILD)) {
333
0
            return;
334
0
          } else {
335
0
            zend_clear_exception();
336
0
          }
337
0
        }
338
12
        return /* self */;
339
12
      case RS_SELF:
340
9
        if (object->nextElement && (object->mode == RIT_SELF_FIRST || object->mode == RIT_CHILD_FIRST)) {
341
0
          zend_call_method_with_0_params(Z_OBJ_P(zthis), object->ce, &object->nextElement, "nextelement", NULL);
342
0
        }
343
9
        if (object->mode == RIT_SELF_FIRST) {
344
9
          object->iterators[object->level].state = RS_CHILD;
345
9
        } else {
346
0
          object->iterators[object->level].state = RS_NEXT;
347
0
        }
348
9
        return /* self */;
349
9
      case RS_CHILD:
350
9
        if (object->callGetChildren) {
351
0
          zend_call_method_with_0_params(Z_OBJ_P(zthis), object->ce, &object->callGetChildren, "callGetChildren", &child);
352
9
        } else {
353
9
          zend_class_entry *ce = object->iterators[object->level].ce;
354
9
          zend_object *obj = Z_OBJ(object->iterators[object->level].zobject);
355
9
          zend_function **cache = &object->iterators[object->level].getchildren;
356
357
9
          zend_call_method_with_0_params(obj, ce, cache, "getchildren", &child);
358
9
        }
359
360
9
        if (EG(exception)) {
361
0
          if (!(object->flags & RIT_CATCH_GET_CHILD)) {
362
0
            return;
363
0
          } else {
364
0
            zend_clear_exception();
365
0
            zval_ptr_dtor(&child);
366
0
            object->iterators[object->level].state = RS_NEXT;
367
0
            goto next_step;
368
0
          }
369
0
        }
370
371
9
        if (Z_TYPE(child) == IS_UNDEF || Z_TYPE(child) != IS_OBJECT ||
372
9
            !((ce = Z_OBJCE(child)) && instanceof_function(ce, spl_ce_RecursiveIterator))) {
373
0
          zval_ptr_dtor(&child);
374
0
          zend_throw_exception(spl_ce_UnexpectedValueException, "Objects returned by RecursiveIterator::getChildren() must implement RecursiveIterator", 0);
375
0
          return;
376
0
        }
377
378
9
        if (object->mode == RIT_CHILD_FIRST) {
379
0
          object->iterators[object->level].state = RS_SELF;
380
9
        } else {
381
9
          object->iterators[object->level].state = RS_NEXT;
382
9
        }
383
9
        object->iterators = erealloc(object->iterators, sizeof(spl_sub_iterator) * (++object->level+1));
384
9
        sub_iter = ce->get_iterator(ce, &child, 0);
385
9
        ZVAL_COPY_VALUE(&object->iterators[object->level].zobject, &child);
386
9
        object->iterators[object->level].iterator = sub_iter;
387
9
        object->iterators[object->level].ce = ce;
388
9
        object->iterators[object->level].state = RS_START;
389
9
        if (object->level > 0
390
9
         && object->iterators[object->level - 1].ce == 0) {
391
0
          object->iterators[object->level].haschildren =
392
0
            object->iterators[object->level - 1].haschildren;
393
0
          object->iterators[object->level].getchildren =
394
0
            object->iterators[object->level - 1].getchildren;
395
9
        } else {
396
9
          object->iterators[object->level].haschildren = NULL;
397
9
          object->iterators[object->level].getchildren = NULL;
398
9
        }
399
9
        if (sub_iter->funcs->rewind) {
400
9
          sub_iter->funcs->rewind(sub_iter);
401
9
        }
402
9
        if (object->beginChildren) {
403
0
          zend_call_method_with_0_params(Z_OBJ_P(zthis), object->ce, &object->beginChildren, "beginchildren", NULL);
404
0
          if (EG(exception)) {
405
0
            if (!(object->flags & RIT_CATCH_GET_CHILD)) {
406
0
              return;
407
0
            } else {
408
0
              zend_clear_exception();
409
0
            }
410
0
          }
411
0
        }
412
9
        goto next_step;
413
60
    }
414
    /* no more elements */
415
21
    if (object->level > 0) {
416
9
      if (object->endChildren) {
417
0
        zend_call_method_with_0_params(Z_OBJ_P(zthis), object->ce, &object->endChildren, "endchildren", NULL);
418
0
        if (EG(exception)) {
419
0
          if (!(object->flags & RIT_CATCH_GET_CHILD)) {
420
0
            return;
421
0
          } else {
422
0
            zend_clear_exception();
423
0
          }
424
0
        }
425
0
      }
426
9
      if (object->level > 0) {
427
9
        zval garbage;
428
9
        ZVAL_COPY_VALUE(&garbage, &object->iterators[object->level].zobject);
429
9
        ZVAL_UNDEF(&object->iterators[object->level].zobject);
430
9
        zval_ptr_dtor(&garbage);
431
9
        zend_iterator_dtor(iterator);
432
9
        object->level--;
433
9
      }
434
12
    } else {
435
12
      return; /* done completeley */
436
12
    }
437
21
  }
438
33
}
439
440
static void spl_recursive_it_rewind_ex(spl_recursive_it_object *object, zval *zthis)
441
12
{
442
12
  zend_object_iterator *sub_iter;
443
444
12
  SPL_FETCH_SUB_ITERATOR(sub_iter, object);
445
446
12
  while (object->level) {
447
0
    sub_iter = object->iterators[object->level].iterator;
448
0
    zend_iterator_dtor(sub_iter);
449
0
    zval_ptr_dtor(&object->iterators[object->level--].zobject);
450
0
    if (!EG(exception) && (!object->endChildren || object->endChildren->common.scope != spl_ce_RecursiveIteratorIterator)) {
451
0
      zend_call_method_with_0_params(Z_OBJ_P(zthis), object->ce, &object->endChildren, "endchildren", NULL);
452
0
    }
453
0
  }
454
12
  object->iterators = erealloc(object->iterators, sizeof(spl_sub_iterator));
455
12
  object->iterators[0].state = RS_START;
456
12
  sub_iter = object->iterators[0].iterator;
457
12
  if (sub_iter->funcs->rewind) {
458
12
    sub_iter->funcs->rewind(sub_iter);
459
12
  }
460
12
  if (!EG(exception) && object->beginIteration && !object->in_iteration) {
461
0
    zend_call_method_with_0_params(Z_OBJ_P(zthis), object->ce, &object->beginIteration, "beginIteration", NULL);
462
0
  }
463
12
  object->in_iteration = true;
464
12
  spl_recursive_it_move_forward_ex(object, zthis);
465
12
}
466
467
static void spl_recursive_it_move_forward(zend_object_iterator *iter)
468
21
{
469
21
  spl_recursive_it_move_forward_ex(Z_SPLRECURSIVE_IT_P(&iter->data), &iter->data);
470
21
}
471
472
static void spl_recursive_it_rewind(zend_object_iterator *iter)
473
12
{
474
12
  spl_recursive_it_rewind_ex(Z_SPLRECURSIVE_IT_P(&iter->data), &iter->data);
475
12
}
476
477
static const zend_object_iterator_funcs spl_recursive_it_iterator_funcs = {
478
  spl_recursive_it_dtor,
479
  spl_recursive_it_valid,
480
  spl_recursive_it_get_current_data,
481
  spl_recursive_it_get_current_key,
482
  spl_recursive_it_move_forward,
483
  spl_recursive_it_rewind,
484
  NULL,
485
  NULL, /* get_gc */
486
};
487
488
static zend_object_iterator *spl_recursive_it_get_iterator(zend_class_entry *ce, zval *zobject, int by_ref)
489
12
{
490
12
  if (by_ref) {
491
0
    zend_throw_error(NULL, "An iterator cannot be used with foreach by reference");
492
0
    return NULL;
493
0
  }
494
495
12
  spl_recursive_it_object *object = Z_SPLRECURSIVE_IT_P(zobject);
496
12
  if (object->iterators == NULL) {
497
0
    zend_throw_error(NULL, "Object is not initialized");
498
0
    return NULL;
499
0
  }
500
501
12
  spl_recursive_it_iterator *iterator = emalloc(sizeof(spl_recursive_it_iterator));
502
12
  zend_iterator_init((zend_object_iterator*)iterator);
503
504
12
  ZVAL_OBJ_COPY(&iterator->intern.data, Z_OBJ_P(zobject));
505
12
  iterator->intern.funcs = &spl_recursive_it_iterator_funcs;
506
12
  return (zend_object_iterator*)iterator;
507
12
}
508
509
0
static zend_result spl_get_iterator_from_aggregate(zval *retval, zend_class_entry *ce, zend_object *obj) {
510
0
  zend_function **getiterator_cache =
511
0
    ce->iterator_funcs_ptr ? &ce->iterator_funcs_ptr->zf_new_iterator : NULL;
512
0
  zend_call_method_with_0_params(obj, ce, getiterator_cache, "getiterator", retval);
513
0
  if (EG(exception)) {
514
0
    return FAILURE;
515
0
  }
516
0
  if (Z_TYPE_P(retval) != IS_OBJECT
517
0
      || !instanceof_function(Z_OBJCE_P(retval), zend_ce_traversable)) {
518
0
    zend_throw_exception_ex(spl_ce_LogicException, 0,
519
0
      "%s::getIterator() must return an object that implements Traversable",
520
0
      ZSTR_VAL(ce->name));
521
0
    zval_ptr_dtor(retval);
522
0
    return FAILURE;
523
0
  }
524
0
  return SUCCESS;
525
0
}
526
527
static void spl_RecursiveIteratorIterator_free_iterators(spl_recursive_it_object *object)
528
1.08k
{
529
1.08k
  if (object->iterators) {
530
24
    while (object->level >= 0) {
531
12
      zend_object_iterator *sub_iter = object->iterators[object->level].iterator;
532
12
      zend_iterator_dtor(sub_iter);
533
12
      zval_ptr_dtor(&object->iterators[object->level].zobject);
534
12
      object->level--;
535
12
    }
536
12
    efree(object->iterators);
537
12
    object->iterators = NULL;
538
12
  }
539
1.08k
}
540
541
static void spl_recursive_it_it_construct(INTERNAL_FUNCTION_PARAMETERS, zend_class_entry *ce_base, zend_class_entry *ce_inner, recursive_it_it_type rit_type)
542
20
{
543
20
  zval *object = ZEND_THIS;
544
20
  spl_recursive_it_object *intern;
545
20
  zval *iterator;
546
20
  zend_class_entry *ce_iterator;
547
20
  zend_long mode, flags;
548
20
  zval caching_it, aggregate_retval;
549
550
20
  switch (rit_type) {
551
4
    case RIT_RecursiveTreeIterator: {
552
4
      zend_long user_caching_it_flags = CIT_CATCH_GET_CHILD;
553
4
      mode = RIT_SELF_FIRST;
554
4
      flags = RTIT_BYPASS_KEY;
555
556
4
      if (zend_parse_parameters(ZEND_NUM_ARGS(), "o|lll", &iterator, &flags, &user_caching_it_flags, &mode) == FAILURE) {
557
4
        RETURN_THROWS();
558
4
      }
559
560
0
      if (instanceof_function(Z_OBJCE_P(iterator), zend_ce_aggregate)) {
561
0
        if (spl_get_iterator_from_aggregate(
562
0
            &aggregate_retval, Z_OBJCE_P(iterator), Z_OBJ_P(iterator)) == FAILURE) {
563
0
          RETURN_THROWS();
564
0
        }
565
0
        iterator = &aggregate_retval;
566
0
      } else {
567
0
        Z_ADDREF_P(iterator);
568
0
      }
569
570
0
      zval params[2];
571
0
      ZVAL_COPY_VALUE(&params[0], iterator);
572
0
      ZVAL_LONG(&params[1], user_caching_it_flags);
573
0
      zend_result is_initialized = object_init_with_constructor(&caching_it, spl_ce_RecursiveCachingIterator, 2, params, NULL);
574
0
      zval_ptr_dtor(&params[0]);
575
0
      if (is_initialized == FAILURE) {
576
0
        RETURN_THROWS();
577
0
      }
578
579
0
      iterator = &caching_it;
580
0
      break;
581
0
    }
582
16
    case RIT_RecursiveIteratorIterator:
583
16
    default: {
584
16
      mode = RIT_LEAVES_ONLY;
585
16
      flags = 0;
586
16
      if (zend_parse_parameters(ZEND_NUM_ARGS(), "o|ll", &iterator, &mode, &flags) == FAILURE) {
587
4
        RETURN_THROWS();
588
4
      }
589
590
12
      if (instanceof_function(Z_OBJCE_P(iterator), zend_ce_aggregate)) {
591
0
        if (spl_get_iterator_from_aggregate(
592
0
            &aggregate_retval, Z_OBJCE_P(iterator), Z_OBJ_P(iterator)) == FAILURE) {
593
0
          RETURN_THROWS();
594
0
        }
595
0
        iterator = &aggregate_retval;
596
12
      } else {
597
12
        Z_ADDREF_P(iterator);
598
12
      }
599
12
      break;
600
12
    }
601
20
  }
602
12
  if (!instanceof_function(Z_OBJCE_P(iterator), spl_ce_RecursiveIterator)) {
603
0
    if (iterator) {
604
0
      zval_ptr_dtor(iterator);
605
0
    }
606
0
    zend_throw_exception(spl_ce_InvalidArgumentException, "An instance of RecursiveIterator or IteratorAggregate creating it is required", 0);
607
0
    return;
608
0
  }
609
610
12
  intern = Z_SPLRECURSIVE_IT_P(object);
611
12
  spl_RecursiveIteratorIterator_free_iterators(intern);
612
12
  intern->iterators = emalloc(sizeof(spl_sub_iterator));
613
12
  intern->level = 0;
614
12
  intern->mode = mode;
615
12
  intern->flags = (int)flags;
616
12
  intern->max_depth = -1;
617
12
  intern->in_iteration = false;
618
12
  intern->ce = Z_OBJCE_P(object);
619
620
12
  intern->beginIteration = zend_hash_str_find_ptr(&intern->ce->function_table, "beginiteration", sizeof("beginiteration") - 1);
621
12
  if (intern->beginIteration->common.scope == ce_base) {
622
12
    intern->beginIteration = NULL;
623
12
  }
624
12
  intern->endIteration = zend_hash_str_find_ptr(&intern->ce->function_table, "enditeration", sizeof("enditeration") - 1);
625
12
  if (intern->endIteration->common.scope == ce_base) {
626
12
    intern->endIteration = NULL;
627
12
  }
628
12
  intern->callHasChildren = zend_hash_str_find_ptr(&intern->ce->function_table, "callhaschildren", sizeof("callHasChildren") - 1);
629
12
  if (intern->callHasChildren->common.scope == ce_base) {
630
12
    intern->callHasChildren = NULL;
631
12
  }
632
12
  intern->callGetChildren = zend_hash_str_find_ptr(&intern->ce->function_table, "callgetchildren", sizeof("callGetChildren") - 1);
633
12
  if (intern->callGetChildren->common.scope == ce_base) {
634
12
    intern->callGetChildren = NULL;
635
12
  }
636
12
  intern->beginChildren = zend_hash_str_find_ptr(&intern->ce->function_table, "beginchildren", sizeof("beginchildren") - 1);
637
12
  if (intern->beginChildren->common.scope == ce_base) {
638
12
    intern->beginChildren = NULL;
639
12
  }
640
12
  intern->endChildren = zend_hash_str_find_ptr(&intern->ce->function_table, "endchildren", sizeof("endchildren") - 1);
641
12
  if (intern->endChildren->common.scope == ce_base) {
642
12
    intern->endChildren = NULL;
643
12
  }
644
12
  intern->nextElement = zend_hash_str_find_ptr(&intern->ce->function_table, "nextelement", sizeof("nextElement") - 1);
645
12
  if (intern->nextElement->common.scope == ce_base) {
646
12
    intern->nextElement = NULL;
647
12
  }
648
649
12
  ce_iterator = Z_OBJCE_P(iterator); /* respect inheritance, don't use spl_ce_RecursiveIterator */
650
12
  intern->iterators[0].iterator = ce_iterator->get_iterator(ce_iterator, iterator, 0);
651
12
  ZVAL_OBJ(&intern->iterators[0].zobject, Z_OBJ_P(iterator));
652
12
  intern->iterators[0].ce = ce_iterator;
653
12
  intern->iterators[0].state = RS_START;
654
12
  intern->iterators[0].haschildren = NULL;
655
12
  intern->iterators[0].getchildren = NULL;
656
657
12
  if (EG(exception)) {
658
0
    spl_RecursiveIteratorIterator_free_iterators(intern);
659
0
  }
660
12
}
661
662
/* {{{ Creates a RecursiveIteratorIterator from a RecursiveIterator. */
663
PHP_METHOD(RecursiveIteratorIterator, __construct)
664
16
{
665
16
  spl_recursive_it_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_RecursiveIteratorIterator, zend_ce_iterator, RIT_RecursiveIteratorIterator);
666
16
} /* }}} */
667
668
/* {{{ Rewind the iterator to the first element of the top level inner iterator. */
669
PHP_METHOD(RecursiveIteratorIterator, rewind)
670
0
{
671
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
672
673
0
  ZEND_PARSE_PARAMETERS_NONE();
674
0
  spl_recursive_it_rewind_ex(object, ZEND_THIS);
675
0
} /* }}} */
676
677
/* {{{ Check whether the current position is valid */
678
PHP_METHOD(RecursiveIteratorIterator, valid)
679
0
{
680
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
681
682
0
  ZEND_PARSE_PARAMETERS_NONE();
683
0
  RETURN_BOOL(spl_recursive_it_valid_ex(object, ZEND_THIS) == SUCCESS);
684
0
} /* }}} */
685
686
/* {{{ Access the current key */
687
PHP_METHOD(RecursiveIteratorIterator, key)
688
0
{
689
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
690
0
  zend_object_iterator      *iterator;
691
692
0
  ZEND_PARSE_PARAMETERS_NONE();
693
694
0
  SPL_FETCH_SUB_ITERATOR(iterator, object);
695
696
0
  if (iterator->funcs->get_current_key) {
697
0
    iterator->funcs->get_current_key(iterator, return_value);
698
0
  } else {
699
0
    RETURN_NULL();
700
0
  }
701
0
} /* }}} */
702
703
/* {{{ Access the current element value */
704
PHP_METHOD(RecursiveIteratorIterator, current)
705
0
{
706
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
707
0
  zend_object_iterator      *iterator;
708
0
  zval                      *data;
709
710
0
  ZEND_PARSE_PARAMETERS_NONE();
711
712
0
  SPL_FETCH_SUB_ITERATOR(iterator, object);
713
714
0
  data = iterator->funcs->get_current_data(iterator);
715
0
  if (data) {
716
0
    RETURN_COPY_DEREF(data);
717
0
  }
718
0
} /* }}} */
719
720
/* {{{ Move forward to the next element */
721
PHP_METHOD(RecursiveIteratorIterator, next)
722
0
{
723
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
724
725
0
  ZEND_PARSE_PARAMETERS_NONE();
726
0
  spl_recursive_it_move_forward_ex(object, ZEND_THIS);
727
0
} /* }}} */
728
729
/* {{{ Get the current depth of the recursive iteration */
730
PHP_METHOD(RecursiveIteratorIterator, getDepth)
731
0
{
732
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
733
734
0
  ZEND_PARSE_PARAMETERS_NONE();
735
0
  RETURN_LONG(object->level);
736
0
} /* }}} */
737
738
/* {{{ The current active sub iterator or the iterator at specified level */
739
PHP_METHOD(RecursiveIteratorIterator, getSubIterator)
740
0
{
741
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
742
0
  zend_long level;
743
0
  bool level_is_null = true;
744
0
  zval *value;
745
746
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "|l!", &level, &level_is_null) == FAILURE) {
747
0
    RETURN_THROWS();
748
0
  }
749
750
0
  if (level_is_null) {
751
0
    level = object->level;
752
0
  } else if (level < 0 || level > object->level) {
753
0
    RETURN_NULL();
754
0
  }
755
756
0
  if(!object->iterators) {
757
0
    zend_throw_error(NULL, "The object is in an invalid state as the parent constructor was not called");
758
0
    RETURN_THROWS();
759
0
  }
760
761
0
  value = &object->iterators[level].zobject;
762
0
  RETURN_COPY_DEREF(value);
763
0
} /* }}} */
764
765
/* {{{ The current active sub iterator */
766
PHP_METHOD(RecursiveIteratorIterator, getInnerIterator)
767
0
{
768
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
769
0
  zval      *zobject;
770
771
0
  ZEND_PARSE_PARAMETERS_NONE();
772
0
  SPL_FETCH_SUB_ELEMENT_ADDR(zobject, object, zobject);
773
0
  RETURN_COPY_DEREF(zobject);
774
0
} /* }}} */
775
776
/* {{{ Called when iteration begins (after first rewind() call) */
777
PHP_METHOD(RecursiveIteratorIterator, beginIteration)
778
0
{
779
0
  ZEND_PARSE_PARAMETERS_NONE();
780
  /* nothing to do */
781
0
} /* }}} */
782
783
/* {{{ Called when iteration ends (when valid() first returns false */
784
PHP_METHOD(RecursiveIteratorIterator, endIteration)
785
0
{
786
0
  ZEND_PARSE_PARAMETERS_NONE();
787
  /* nothing to do */
788
0
} /* }}} */
789
790
/* {{{ Called for each element to test whether it has children */
791
PHP_METHOD(RecursiveIteratorIterator, callHasChildren)
792
0
{
793
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
794
0
  zend_class_entry *ce;
795
0
  zval *zobject;
796
797
0
  ZEND_PARSE_PARAMETERS_NONE();
798
799
0
  if (!object->iterators) {
800
0
    RETURN_FALSE;
801
0
  }
802
803
0
  SPL_FETCH_SUB_ELEMENT(ce, object, ce);
804
805
0
  zobject = &object->iterators[object->level].zobject;
806
0
  if (Z_TYPE_P(zobject) == IS_UNDEF) {
807
0
    RETURN_FALSE;
808
0
  } else {
809
0
    zend_call_method_with_0_params(Z_OBJ_P(zobject), ce, &object->iterators[object->level].haschildren, "haschildren", return_value);
810
0
    if (Z_TYPE_P(return_value) == IS_UNDEF) {
811
0
      RETURN_FALSE;
812
0
    }
813
0
  }
814
0
} /* }}} */
815
816
/* {{{ Return children of current element */
817
PHP_METHOD(RecursiveIteratorIterator, callGetChildren)
818
0
{
819
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
820
0
  zend_class_entry *ce;
821
0
  zval *zobject;
822
823
0
  ZEND_PARSE_PARAMETERS_NONE();
824
825
0
  SPL_FETCH_SUB_ELEMENT(ce, object, ce);
826
827
0
  zobject = &object->iterators[object->level].zobject;
828
0
  if (Z_TYPE_P(zobject) == IS_UNDEF) {
829
0
    RETURN_NULL();
830
0
  } else {
831
0
    zend_call_method_with_0_params(Z_OBJ_P(zobject), ce, &object->iterators[object->level].getchildren, "getchildren", return_value);
832
0
    if (Z_TYPE_P(return_value) == IS_UNDEF) {
833
0
      RETURN_NULL();
834
0
    }
835
0
  }
836
0
} /* }}} */
837
838
/* {{{ Called when recursing one level down */
839
PHP_METHOD(RecursiveIteratorIterator, beginChildren)
840
0
{
841
0
  ZEND_PARSE_PARAMETERS_NONE();
842
  /* nothing to do */
843
0
} /* }}} */
844
845
/* {{{ Called when end recursing one level */
846
PHP_METHOD(RecursiveIteratorIterator, endChildren)
847
0
{
848
0
  ZEND_PARSE_PARAMETERS_NONE();
849
  /* nothing to do */
850
0
} /* }}} */
851
852
/* {{{ Called when the next element is available */
853
PHP_METHOD(RecursiveIteratorIterator, nextElement)
854
0
{
855
0
  ZEND_PARSE_PARAMETERS_NONE();
856
  /* nothing to do */
857
0
} /* }}} */
858
859
/* {{{ Set the maximum allowed depth (or any depth if pmax_depth = -1] */
860
PHP_METHOD(RecursiveIteratorIterator, setMaxDepth)
861
0
{
862
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
863
0
  zend_long  max_depth = -1;
864
865
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "|l", &max_depth) == FAILURE) {
866
0
    RETURN_THROWS();
867
0
  }
868
0
  if (max_depth < -1) {
869
0
    zend_argument_value_error(1, "must be greater than or equal to -1");
870
0
    RETURN_THROWS();
871
0
  } else if (max_depth > INT_MAX) {
872
0
    max_depth = INT_MAX;
873
0
  }
874
875
0
  object->max_depth = (int)max_depth;
876
0
} /* }}} */
877
878
/* {{{ Return the maximum accepted depth or false if any depth is allowed */
879
PHP_METHOD(RecursiveIteratorIterator, getMaxDepth)
880
0
{
881
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
882
883
0
  ZEND_PARSE_PARAMETERS_NONE();
884
885
0
  if (object->max_depth == -1) {
886
0
    RETURN_FALSE;
887
0
  } else {
888
0
    RETURN_LONG(object->max_depth);
889
0
  }
890
0
} /* }}} */
891
892
static zend_function *spl_recursive_it_get_method(zend_object **zobject, zend_string *method, const zval *key)
893
0
{
894
0
  zend_function           *function_handler;
895
0
  spl_recursive_it_object *object = spl_recursive_it_from_obj(*zobject);
896
0
  zend_long                     level = object->level;
897
0
  zval                    *zobj;
898
899
0
  if (!object->iterators) {
900
0
    zend_throw_error(NULL, "The %s instance wasn't initialized properly", ZSTR_VAL((*zobject)->ce->name));
901
0
    return NULL;
902
0
  }
903
0
  zobj = &object->iterators[level].zobject;
904
905
0
  function_handler = zend_std_get_method(zobject, method, key);
906
0
  if (!function_handler) {
907
0
    if ((function_handler = zend_hash_find_ptr(&Z_OBJCE_P(zobj)->function_table, method)) == NULL) {
908
0
      *zobject = Z_OBJ_P(zobj);
909
0
      function_handler = (*zobject)->handlers->get_method(zobject, method, key);
910
0
    } else {
911
0
      *zobject = Z_OBJ_P(zobj);
912
0
    }
913
0
  }
914
0
  return function_handler;
915
0
}
916
917
/* {{{ spl_RecursiveIteratorIterator_free_storage */
918
static void spl_RecursiveIteratorIterator_free_storage(zend_object *_object)
919
1.07k
{
920
1.07k
  spl_recursive_it_object *object = spl_recursive_it_from_obj(_object);
921
922
1.07k
  spl_RecursiveIteratorIterator_free_iterators(object);
923
924
1.07k
  zend_object_std_dtor(&object->std);
925
7.53k
  for (size_t i = 0; i < 6; i++) {
926
6.45k
    if (object->prefix[i]) {
927
6.00k
      zend_string_release(object->prefix[i]);
928
6.00k
    }
929
6.45k
  }
930
931
1.07k
  if (object->postfix[0]) {
932
1.00k
    zend_string_release(object->postfix[0]);
933
1.00k
  }
934
1.07k
}
935
/* }}} */
936
937
static HashTable *spl_RecursiveIteratorIterator_get_gc(zend_object *obj, zval **table, int *n)
938
195
{
939
195
  spl_recursive_it_object *object = spl_recursive_it_from_obj(obj);
940
195
  zend_get_gc_buffer *gc_buffer = zend_get_gc_buffer_create();
941
942
195
  if (object->iterators) {
943
48
    for (int level = 0; level <= object->level; level++) {
944
24
      zend_get_gc_buffer_add_zval(gc_buffer, &object->iterators[level].zobject);
945
24
      zend_get_gc_buffer_add_obj(gc_buffer, &object->iterators[level].iterator->std);
946
24
    }
947
24
  }
948
949
195
  zend_get_gc_buffer_use(gc_buffer, table, n);
950
195
  return zend_std_get_properties(obj);
951
195
}
952
953
/* {{{ spl_RecursiveIteratorIterator_new_ex */
954
static zend_object *spl_RecursiveIteratorIterator_new_ex(zend_class_entry *class_type, int init_prefix)
955
1.07k
{
956
1.07k
  spl_recursive_it_object *intern;
957
958
1.07k
  intern = zend_object_alloc(sizeof(spl_recursive_it_object), class_type);
959
960
1.07k
  if (init_prefix) {
961
1.00k
    intern->prefix[0] = ZSTR_EMPTY_ALLOC();
962
1.00k
    intern->prefix[1] = ZSTR_INIT_LITERAL("| ", 0);
963
1.00k
    intern->prefix[2] = ZSTR_INIT_LITERAL("  ", 0);
964
1.00k
    intern->prefix[3] = ZSTR_INIT_LITERAL("|-", 0);
965
1.00k
    intern->prefix[4] = ZSTR_INIT_LITERAL("\\-", 0);
966
1.00k
    intern->prefix[5] = ZSTR_EMPTY_ALLOC();
967
968
1.00k
    intern->postfix[0] = ZSTR_EMPTY_ALLOC();
969
1.00k
  }
970
971
1.07k
  zend_object_std_init(&intern->std, class_type);
972
1.07k
  object_properties_init(&intern->std, class_type);
973
974
1.07k
  return &intern->std;
975
1.07k
}
976
/* }}} */
977
978
/* {{{ spl_RecursiveIteratorIterator_new */
979
static zend_object *spl_RecursiveIteratorIterator_new(zend_class_entry *class_type)
980
76
{
981
76
  return spl_RecursiveIteratorIterator_new_ex(class_type, 0);
982
76
}
983
/* }}} */
984
985
/* {{{ spl_RecursiveTreeIterator_new */
986
static zend_object *spl_RecursiveTreeIterator_new(zend_class_entry *class_type)
987
1.00k
{
988
1.00k
  return spl_RecursiveIteratorIterator_new_ex(class_type, 1);
989
1.00k
}
990
/* }}} */
991
992
static zend_string *spl_recursive_tree_iterator_get_prefix(spl_recursive_it_object *object)
993
0
{
994
0
  smart_str  str = {0};
995
0
  zval       has_next;
996
0
  int        level;
997
998
0
  smart_str_append(&str, object->prefix[0]);
999
1000
0
  for (level = 0; level < object->level; ++level) {
1001
0
    zend_call_method_with_0_params(Z_OBJ(object->iterators[level].zobject), object->iterators[level].ce, NULL, "hasnext", &has_next);
1002
0
    if (Z_TYPE(has_next) != IS_UNDEF) {
1003
0
      if (Z_TYPE(has_next) == IS_TRUE) {
1004
0
        smart_str_append(&str, object->prefix[1]);
1005
0
      } else {
1006
0
        smart_str_append(&str, object->prefix[2]);
1007
0
      }
1008
0
      zval_ptr_dtor(&has_next);
1009
0
    }
1010
0
  }
1011
0
  zend_call_method_with_0_params(Z_OBJ(object->iterators[level].zobject), object->iterators[level].ce, NULL, "hasnext", &has_next);
1012
0
  if (Z_TYPE(has_next) != IS_UNDEF) {
1013
0
    if (Z_TYPE(has_next) == IS_TRUE) {
1014
0
      smart_str_append(&str, object->prefix[3]);
1015
0
    } else {
1016
0
      smart_str_append(&str, object->prefix[4]);
1017
0
    }
1018
0
    zval_ptr_dtor(&has_next);
1019
0
  }
1020
1021
0
  smart_str_append(&str, object->prefix[5]);
1022
0
  smart_str_0(&str);
1023
1024
0
  return str.s;
1025
0
}
1026
1027
static zend_string *spl_recursive_tree_iterator_get_entry(spl_recursive_it_object *object)
1028
0
{
1029
0
  zend_object_iterator *iterator = object->iterators[object->level].iterator;
1030
0
  zval *data = iterator->funcs->get_current_data(iterator);
1031
0
  if (!data) {
1032
0
    return NULL;
1033
0
  }
1034
1035
0
  ZVAL_DEREF(data);
1036
0
  if (Z_TYPE_P(data) == IS_ARRAY) {
1037
    /* TODO: Remove this special case? */
1038
0
    return ZSTR_KNOWN(ZEND_STR_ARRAY_CAPITALIZED);
1039
0
  }
1040
0
  return zval_get_string(data);
1041
0
}
1042
1043
static zend_string *spl_recursive_tree_iterator_get_postfix(spl_recursive_it_object *object)
1044
0
{
1045
0
  return zend_string_copy(object->postfix[0]);
1046
0
}
1047
1048
/* {{{ RecursiveIteratorIterator to generate ASCII graphic trees for the entries in a RecursiveIterator */
1049
PHP_METHOD(RecursiveTreeIterator, __construct)
1050
4
{
1051
4
  spl_recursive_it_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_RecursiveTreeIterator, zend_ce_iterator, RIT_RecursiveTreeIterator);
1052
4
} /* }}} */
1053
1054
/* {{{ Sets prefix parts as used in getPrefix() */
1055
PHP_METHOD(RecursiveTreeIterator, setPrefixPart)
1056
0
{
1057
0
  zend_long  part;
1058
0
  zend_string *prefix;
1059
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
1060
1061
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "lS", &part, &prefix) == FAILURE) {
1062
0
    RETURN_THROWS();
1063
0
  }
1064
1065
0
  if (0 > part || part > 5) {
1066
0
    zend_argument_value_error(1, "must be a RecursiveTreeIterator::PREFIX_* constant");
1067
0
    RETURN_THROWS();
1068
0
  }
1069
1070
0
  zend_string_release(object->prefix[part]);
1071
0
  object->prefix[part] = zend_string_copy(prefix);
1072
0
} /* }}} */
1073
1074
/* {{{ Returns the string to place in front of current element */
1075
PHP_METHOD(RecursiveTreeIterator, getPrefix)
1076
0
{
1077
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
1078
1079
0
  ZEND_PARSE_PARAMETERS_NONE();
1080
1081
0
  if(!object->iterators) {
1082
0
    zend_throw_error(NULL, "The object is in an invalid state as the parent constructor was not called");
1083
0
    RETURN_THROWS();
1084
0
  }
1085
1086
0
  RETURN_STR(spl_recursive_tree_iterator_get_prefix(object));
1087
0
} /* }}} */
1088
1089
/* {{{ Sets postfix as used in getPostfix() */
1090
PHP_METHOD(RecursiveTreeIterator, setPostfix)
1091
0
{
1092
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
1093
0
  zend_string *postfix;
1094
1095
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "S", &postfix) == FAILURE) {
1096
0
    RETURN_THROWS();
1097
0
  }
1098
1099
0
  zend_string_release(object->postfix[0]);
1100
0
  object->postfix[0] = zend_string_copy(postfix);
1101
0
} /* }}} */
1102
1103
/* {{{ Returns the string presentation built for current element */
1104
PHP_METHOD(RecursiveTreeIterator, getEntry)
1105
0
{
1106
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
1107
1108
0
  ZEND_PARSE_PARAMETERS_NONE();
1109
1110
0
  if(!object->iterators) {
1111
0
    zend_throw_error(NULL, "The object is in an invalid state as the parent constructor was not called");
1112
0
    RETURN_THROWS();
1113
0
  }
1114
1115
0
  zend_string *entry = spl_recursive_tree_iterator_get_entry(object);
1116
0
  if (!entry) {
1117
    // TODO: Can this happen? It's not in the stubs.
1118
0
    RETURN_NULL();
1119
0
  }
1120
0
  RETURN_STR(entry);
1121
0
} /* }}} */
1122
1123
/* {{{ Returns the string to place after the current element */
1124
PHP_METHOD(RecursiveTreeIterator, getPostfix)
1125
0
{
1126
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
1127
1128
0
  ZEND_PARSE_PARAMETERS_NONE();
1129
1130
0
  if(!object->iterators) {
1131
0
    zend_throw_error(NULL, "The object is in an invalid state as the parent constructor was not called");
1132
0
    RETURN_THROWS();
1133
0
  }
1134
1135
0
  RETURN_STR(spl_recursive_tree_iterator_get_postfix(object));
1136
0
} /* }}} */
1137
1138
/* {{{ Returns the current element prefixed and postfixed */
1139
PHP_METHOD(RecursiveTreeIterator, current)
1140
0
{
1141
0
  spl_recursive_it_object *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
1142
1143
0
  ZEND_PARSE_PARAMETERS_NONE();
1144
1145
0
  if(!object->iterators) {
1146
0
    zend_throw_error(NULL, "The object is in an invalid state as the parent constructor was not called");
1147
0
    RETURN_THROWS();
1148
0
  }
1149
1150
0
  if (object->flags & RTIT_BYPASS_CURRENT) {
1151
0
    zend_object_iterator      *iterator;
1152
0
    zval                      *data;
1153
1154
0
    SPL_FETCH_SUB_ITERATOR(iterator, object);
1155
0
    data = iterator->funcs->get_current_data(iterator);
1156
0
    if (data) {
1157
0
      RETURN_COPY_DEREF(data);
1158
0
    } else {
1159
0
      RETURN_NULL();
1160
0
    }
1161
0
  }
1162
1163
0
  zend_string *entry = spl_recursive_tree_iterator_get_entry(object);
1164
0
  if (!entry) {
1165
0
    RETURN_NULL();
1166
0
  }
1167
1168
0
  zend_string *prefix = spl_recursive_tree_iterator_get_prefix(object);
1169
0
  zend_string *postfix = spl_recursive_tree_iterator_get_postfix(object);
1170
1171
0
  zend_string *result = zend_string_concat3(
1172
0
    ZSTR_VAL(prefix), ZSTR_LEN(prefix),
1173
0
    ZSTR_VAL(entry), ZSTR_LEN(entry),
1174
0
    ZSTR_VAL(postfix), ZSTR_LEN(postfix));
1175
1176
0
  zend_string_release(entry);
1177
0
  zend_string_release(prefix);
1178
0
  zend_string_release(postfix);
1179
1180
0
  RETURN_NEW_STR(result);
1181
0
} /* }}} */
1182
1183
/* {{{ Returns the current key prefixed and postfixed */
1184
PHP_METHOD(RecursiveTreeIterator, key)
1185
0
{
1186
0
  spl_recursive_it_object   *object = Z_SPLRECURSIVE_IT_P(ZEND_THIS);
1187
0
  zend_object_iterator      *iterator;
1188
0
  zval                       key;
1189
1190
0
  ZEND_PARSE_PARAMETERS_NONE();
1191
1192
0
  SPL_FETCH_SUB_ITERATOR(iterator, object);
1193
1194
0
  if (iterator->funcs->get_current_key) {
1195
0
    iterator->funcs->get_current_key(iterator, &key);
1196
0
  } else {
1197
0
    ZVAL_NULL(&key);
1198
0
  }
1199
1200
0
  if (object->flags & RTIT_BYPASS_KEY) {
1201
0
    RETURN_COPY_VALUE(&key);
1202
0
  }
1203
1204
0
  zend_string *key_str = zval_get_string(&key);
1205
0
  zend_string *prefix = spl_recursive_tree_iterator_get_prefix(object);
1206
0
  zend_string *postfix = spl_recursive_tree_iterator_get_postfix(object);
1207
1208
0
  zend_string *result = zend_string_concat3(
1209
0
    ZSTR_VAL(prefix), ZSTR_LEN(prefix),
1210
0
    ZSTR_VAL(key_str), ZSTR_LEN(key_str),
1211
0
    ZSTR_VAL(postfix), ZSTR_LEN(postfix));
1212
1213
0
  zend_string_release(key_str);
1214
0
  zend_string_release(prefix);
1215
0
  zend_string_release(postfix);
1216
0
  zval_ptr_dtor(&key);
1217
1218
0
  RETURN_NEW_STR(result);
1219
0
} /* }}} */
1220
1221
static zend_function *spl_dual_it_get_method(zend_object **object, zend_string *method, const zval *key)
1222
131
{
1223
131
  zend_function        *function_handler;
1224
131
  spl_dual_it_object   *intern;
1225
1226
131
  intern = spl_dual_it_from_obj(*object);
1227
1228
131
  function_handler = zend_std_get_method(object, method, key);
1229
131
  if (!function_handler && intern->inner.ce) {
1230
91
    if ((function_handler = zend_hash_find_ptr(&intern->inner.ce->function_table, method)) == NULL) {
1231
91
      if (Z_OBJ_HT(intern->inner.zobject)->get_method) {
1232
91
        *object = Z_OBJ(intern->inner.zobject);
1233
91
        function_handler = (*object)->handlers->get_method(object, method, key);
1234
91
      }
1235
91
    } else {
1236
0
      *object = Z_OBJ(intern->inner.zobject);
1237
0
    }
1238
91
  }
1239
131
  return function_handler;
1240
131
}
1241
1242
#define SPL_CHECK_CTOR(intern, classname) \
1243
0
  if (intern->dit_type == DIT_Unknown) { \
1244
0
    /* TODO Normal Error? */ \
1245
0
    zend_throw_exception_ex(spl_ce_BadMethodCallException, 0, "Classes derived from %s must call %s::__construct()", \
1246
0
        ZSTR_VAL((spl_ce_##classname)->name), ZSTR_VAL((spl_ce_##classname)->name)); \
1247
0
    RETURN_THROWS(); \
1248
0
  }
1249
1250
0
#define APPENDIT_CHECK_CTOR(intern) SPL_CHECK_CTOR(intern, AppendIterator)
1251
1252
static inline zend_result spl_dual_it_fetch(spl_dual_it_object *intern, int check_more);
1253
1254
static inline zend_result spl_cit_check_flags(zend_long flags)
1255
0
{
1256
0
  zend_long cnt = 0;
1257
1258
0
  cnt += (flags & CIT_CALL_TOSTRING) ? 1 : 0;
1259
0
  cnt += (flags & CIT_TOSTRING_USE_KEY) ? 1 : 0;
1260
0
  cnt += (flags & CIT_TOSTRING_USE_CURRENT) ? 1 : 0;
1261
0
  cnt += (flags & CIT_TOSTRING_USE_INNER) ? 1 : 0;
1262
1263
0
  return cnt <= 1 ? SUCCESS : FAILURE;
1264
0
}
1265
1266
static spl_dual_it_object* spl_dual_it_construct(INTERNAL_FUNCTION_PARAMETERS, zend_class_entry *ce_base, zend_class_entry *ce_inner, dual_it_type dit_type)
1267
155
{
1268
155
  zval                 *zobject, retval;
1269
155
  spl_dual_it_object   *intern;
1270
155
  zend_class_entry     *ce = NULL;
1271
155
  int                   inc_refcount = 1;
1272
155
  zend_error_handling   error_handling;
1273
1274
155
  intern = Z_SPLDUAL_IT_P(ZEND_THIS);
1275
1276
155
  if (intern->dit_type != DIT_Unknown) {
1277
0
    zend_throw_exception_ex(spl_ce_BadMethodCallException, 0, "%s::getIterator() must be called exactly once per instance", ZSTR_VAL(ce_base->name));
1278
0
    return NULL;
1279
0
  }
1280
1281
155
  switch (dit_type) {
1282
5
    case DIT_LimitIterator: {
1283
5
      intern->u.limit.offset = 0; /* start at beginning */
1284
5
      intern->u.limit.count = -1; /* get all */
1285
5
      if (zend_parse_parameters(ZEND_NUM_ARGS(), "O|ll", &zobject, ce_inner, &intern->u.limit.offset, &intern->u.limit.count) == FAILURE) {
1286
4
        return NULL;
1287
4
      }
1288
1
      if (intern->u.limit.offset < 0) {
1289
0
        zend_argument_value_error(2, "must be greater than or equal to 0");
1290
0
        return NULL;
1291
0
      }
1292
1
      if (intern->u.limit.count < -1) {
1293
0
        zend_argument_value_error(3, "must be greater than or equal to -1");
1294
0
        return NULL;
1295
0
      }
1296
1
      break;
1297
1
    }
1298
4
    case DIT_CachingIterator:
1299
8
    case DIT_RecursiveCachingIterator: {
1300
8
      zend_long flags = CIT_CALL_TOSTRING;
1301
8
      if (zend_parse_parameters(ZEND_NUM_ARGS(), "O|l", &zobject, ce_inner, &flags) == FAILURE) {
1302
8
        return NULL;
1303
8
      }
1304
0
      if (spl_cit_check_flags(flags) != SUCCESS) {
1305
0
        zend_argument_value_error(2, "must contain only one of CachingIterator::CALL_TOSTRING, "
1306
0
          "CachingIterator::TOSTRING_USE_KEY, CachingIterator::TOSTRING_USE_CURRENT, "
1307
0
          "or CachingIterator::TOSTRING_USE_INNER");
1308
0
        return NULL;
1309
0
      }
1310
0
      intern->u.caching.flags |= flags & CIT_PUBLIC;
1311
0
      array_init(&intern->u.caching.zcache);
1312
0
      break;
1313
0
    }
1314
53
    case DIT_IteratorIterator: {
1315
53
      zend_class_entry *ce_cast;
1316
53
      zend_string *class_name = NULL;
1317
1318
53
      if (zend_parse_parameters(ZEND_NUM_ARGS(), "O|S!", &zobject, ce_inner, &class_name) == FAILURE) {
1319
4
        return NULL;
1320
4
      }
1321
49
      ce = Z_OBJCE_P(zobject);
1322
49
      if (!instanceof_function(ce, zend_ce_iterator)) {
1323
0
        if (class_name) {
1324
0
          if (!(ce_cast = zend_lookup_class(class_name))
1325
0
          || !instanceof_function(ce, ce_cast)
1326
0
          || !ce_cast->get_iterator
1327
0
          ) {
1328
0
            zend_throw_exception(spl_ce_LogicException, "Class to downcast to not found or not base class or does not implement Traversable", 0);
1329
0
            return NULL;
1330
0
          }
1331
0
          ce = ce_cast;
1332
0
        }
1333
0
        if (instanceof_function(ce, zend_ce_aggregate)) {
1334
0
          if (spl_get_iterator_from_aggregate(&retval, ce, Z_OBJ_P(zobject)) == FAILURE) {
1335
0
            return NULL;
1336
0
          }
1337
0
          zobject = &retval;
1338
0
          ce = Z_OBJCE_P(zobject);
1339
0
          inc_refcount = 0;
1340
0
        }
1341
0
      }
1342
49
      break;
1343
49
    }
1344
49
    case DIT_RegexIterator:
1345
8
    case DIT_RecursiveRegexIterator: {
1346
8
      zend_string *regex;
1347
8
      zend_long mode = REGIT_MODE_MATCH;
1348
1349
8
      intern->u.regex.flags = 0;
1350
8
      intern->u.regex.preg_flags = 0;
1351
8
      if (zend_parse_parameters(ZEND_NUM_ARGS(), "OS|lll", &zobject, ce_inner, &regex, &mode, &intern->u.regex.flags, &intern->u.regex.preg_flags) == FAILURE) {
1352
8
        return NULL;
1353
8
      }
1354
0
      if (mode < 0 || mode >= REGIT_MODE_MAX) {
1355
0
        zend_argument_value_error(3, "must be RegexIterator::MATCH, RegexIterator::GET_MATCH, "
1356
0
          "RegexIterator::ALL_MATCHES, RegexIterator::SPLIT, or RegexIterator::REPLACE");
1357
0
        return NULL;
1358
0
      }
1359
1360
      /* pcre_get_compiled_regex_cache() might emit E_WARNINGs that we want to promote to exception */
1361
0
      zend_replace_error_handling(EH_THROW, spl_ce_InvalidArgumentException, &error_handling);
1362
0
      intern->u.regex.pce = pcre_get_compiled_regex_cache(regex);
1363
0
      zend_restore_error_handling(&error_handling);
1364
1365
0
      if (intern->u.regex.pce == NULL) {
1366
        /* pcre_get_compiled_regex_cache has already sent error */
1367
0
        return NULL;
1368
0
      }
1369
0
      intern->u.regex.mode = mode;
1370
0
      intern->u.regex.regex = zend_string_copy(regex);
1371
0
      php_pcre_pce_incref(intern->u.regex.pce);
1372
0
      break;
1373
0
    }
1374
69
    case DIT_CallbackFilterIterator:
1375
73
    case DIT_RecursiveCallbackFilterIterator: {
1376
73
      zend_fcall_info fci;
1377
73
      if (zend_parse_parameters(ZEND_NUM_ARGS(), "OF", &zobject, ce_inner, &fci, &intern->u.callback_filter) == FAILURE) {
1378
13
        return NULL;
1379
13
      }
1380
60
      zend_fcc_addref(&intern->u.callback_filter);
1381
60
      break;
1382
73
    }
1383
8
    default:
1384
8
      if (zend_parse_parameters(ZEND_NUM_ARGS(), "O", &zobject, ce_inner) == FAILURE) {
1385
8
        return NULL;
1386
8
      }
1387
0
      break;
1388
155
  }
1389
1390
110
  intern->dit_type = dit_type;
1391
110
  if (inc_refcount) {
1392
110
    Z_ADDREF_P(zobject);
1393
110
  }
1394
110
  ZVAL_OBJ(&intern->inner.zobject, Z_OBJ_P(zobject));
1395
1396
110
  intern->inner.ce = dit_type == DIT_IteratorIterator ? ce : Z_OBJCE_P(zobject);
1397
110
  intern->inner.object = Z_OBJ_P(zobject);
1398
110
  intern->inner.iterator = intern->inner.ce->get_iterator(intern->inner.ce, zobject, 0);
1399
1400
110
  return intern;
1401
155
}
1402
1403
/* {{{ Create an Iterator from another iterator */
1404
PHP_METHOD(FilterIterator, __construct)
1405
0
{
1406
0
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_FilterIterator, zend_ce_iterator, DIT_FilterIterator);
1407
0
} /* }}} */
1408
1409
/* {{{ Create an Iterator from another iterator */
1410
PHP_METHOD(CallbackFilterIterator, __construct)
1411
69
{
1412
69
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_CallbackFilterIterator, zend_ce_iterator, DIT_CallbackFilterIterator);
1413
69
} /* }}} */
1414
1415
/* {{{ Get the inner iterator */
1416
PHP_METHOD(IteratorIterator, getInnerIterator)
1417
12
{
1418
12
  spl_dual_it_object   *intern;
1419
1420
12
  ZEND_PARSE_PARAMETERS_NONE();
1421
1422
12
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1423
1424
12
  if (!Z_ISUNDEF(intern->inner.zobject)) {
1425
12
    zval *value = &intern->inner.zobject;
1426
12
    RETURN_COPY_DEREF(value);
1427
12
  } else {
1428
0
    RETURN_NULL();
1429
0
  }
1430
12
} /* }}} */
1431
1432
static inline void spl_dual_it_free(spl_dual_it_object *intern)
1433
943
{
1434
943
  if (intern->inner.iterator && intern->inner.iterator->funcs->invalidate_current) {
1435
88
    intern->inner.iterator->funcs->invalidate_current(intern->inner.iterator);
1436
88
  }
1437
943
  if (Z_TYPE(intern->current.data) != IS_UNDEF) {
1438
159
    zval_ptr_dtor(&intern->current.data);
1439
159
    ZVAL_UNDEF(&intern->current.data);
1440
159
  }
1441
943
  if (Z_TYPE(intern->current.key) != IS_UNDEF) {
1442
159
    zval_ptr_dtor(&intern->current.key);
1443
159
    ZVAL_UNDEF(&intern->current.key);
1444
159
  }
1445
943
  if (intern->dit_type == DIT_CachingIterator || intern->dit_type == DIT_RecursiveCachingIterator) {
1446
0
    if (intern->u.caching.zstr) {
1447
0
      zend_string_release(intern->u.caching.zstr);
1448
0
      intern->u.caching.zstr = NULL;
1449
0
    }
1450
0
    if (Z_TYPE(intern->u.caching.zchildren) != IS_UNDEF) {
1451
0
      zval_ptr_dtor(&intern->u.caching.zchildren);
1452
0
      ZVAL_UNDEF(&intern->u.caching.zchildren);
1453
0
    }
1454
0
  }
1455
943
}
1456
1457
static inline void spl_dual_it_rewind(spl_dual_it_object *intern)
1458
108
{
1459
108
  spl_dual_it_free(intern);
1460
108
  intern->current.pos = 0;
1461
108
  if (intern->inner.iterator && intern->inner.iterator->funcs->rewind) {
1462
108
    intern->inner.iterator->funcs->rewind(intern->inner.iterator);
1463
108
  }
1464
108
}
1465
1466
static inline zend_result spl_dual_it_valid(spl_dual_it_object *intern)
1467
210
{
1468
210
  if (!intern->inner.iterator) {
1469
0
    return FAILURE;
1470
0
  }
1471
  /* FAILURE / SUCCESS */
1472
210
  return intern->inner.iterator->funcs->valid(intern->inner.iterator);
1473
210
}
1474
1475
static inline zend_result spl_dual_it_fetch(spl_dual_it_object *intern, int check_more)
1476
210
{
1477
210
  zval *data;
1478
1479
210
  spl_dual_it_free(intern);
1480
210
  if (!check_more || spl_dual_it_valid(intern) == SUCCESS) {
1481
159
    data = intern->inner.iterator->funcs->get_current_data(intern->inner.iterator);
1482
159
    if (data) {
1483
159
      ZVAL_COPY(&intern->current.data, data);
1484
159
    }
1485
1486
159
    if (intern->inner.iterator->funcs->get_current_key) {
1487
159
      intern->inner.iterator->funcs->get_current_key(intern->inner.iterator, &intern->current.key);
1488
159
      if (EG(exception)) {
1489
0
        zval_ptr_dtor(&intern->current.key);
1490
0
        ZVAL_UNDEF(&intern->current.key);
1491
0
      }
1492
159
    } else {
1493
0
      ZVAL_LONG(&intern->current.key, intern->current.pos);
1494
0
    }
1495
159
    return EG(exception) ? FAILURE : SUCCESS;
1496
159
  }
1497
51
  return FAILURE;
1498
210
}
1499
1500
static inline void spl_dual_it_next(spl_dual_it_object *intern, int do_free)
1501
103
{
1502
103
  if (do_free) {
1503
103
    spl_dual_it_free(intern);
1504
103
  } else if (!intern->inner.iterator) {
1505
0
    zend_throw_error(NULL, "The inner constructor wasn't initialized with an iterator instance");
1506
0
    return;
1507
0
  }
1508
103
  intern->inner.iterator->funcs->move_forward(intern->inner.iterator);
1509
103
  intern->current.pos++;
1510
103
}
1511
1512
/* {{{ Rewind the iterator */
1513
PHP_METHOD(IteratorIterator, rewind)
1514
37
{
1515
37
  spl_dual_it_object   *intern;
1516
1517
37
  ZEND_PARSE_PARAMETERS_NONE();
1518
1519
37
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1520
1521
37
  spl_dual_it_rewind(intern);
1522
37
  spl_dual_it_fetch(intern, 1);
1523
37
} /* }}} */
1524
1525
/* {{{ Check whether the current element is valid */
1526
PHP_METHOD(IteratorIterator, valid)
1527
166
{
1528
166
  spl_dual_it_object   *intern;
1529
1530
166
  ZEND_PARSE_PARAMETERS_NONE();
1531
1532
166
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1533
1534
166
  RETURN_BOOL(Z_TYPE(intern->current.data) != IS_UNDEF);
1535
166
} /* }}} */
1536
1537
/* {{{ Get the current key */
1538
PHP_METHOD(IteratorIterator, key)
1539
41
{
1540
41
  spl_dual_it_object   *intern;
1541
1542
41
  ZEND_PARSE_PARAMETERS_NONE();
1543
1544
41
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1545
1546
41
  if (Z_TYPE(intern->current.key) != IS_UNDEF) {
1547
41
    RETURN_COPY_DEREF(&intern->current.key);
1548
41
  } else {
1549
0
    RETURN_NULL();
1550
0
  }
1551
41
} /* }}} */
1552
1553
/* {{{ Get the current element value */
1554
PHP_METHOD(IteratorIterator, current)
1555
125
{
1556
125
  spl_dual_it_object   *intern;
1557
1558
125
  ZEND_PARSE_PARAMETERS_NONE();
1559
1560
125
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1561
1562
125
  if (Z_TYPE(intern->current.data) != IS_UNDEF) {
1563
125
    RETURN_COPY_DEREF(&intern->current.data);
1564
125
  } else {
1565
0
    RETURN_NULL();
1566
0
  }
1567
125
} /* }}} */
1568
1569
/* {{{ Move the iterator forward */
1570
PHP_METHOD(IteratorIterator, next)
1571
83
{
1572
83
  spl_dual_it_object   *intern;
1573
1574
83
  ZEND_PARSE_PARAMETERS_NONE();
1575
1576
83
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1577
1578
83
  spl_dual_it_next(intern, 1);
1579
83
  spl_dual_it_fetch(intern, 1);
1580
83
} /* }}} */
1581
1582
static inline void spl_filter_it_fetch(zval *zthis, spl_dual_it_object *intern)
1583
90
{
1584
90
  zval retval;
1585
1586
90
  while (spl_dual_it_fetch(intern, 1) == SUCCESS) {
1587
65
    zend_call_method_with_0_params(Z_OBJ_P(zthis), intern->std.ce, NULL, "accept", &retval);
1588
65
    if (Z_TYPE(retval) != IS_UNDEF) {
1589
65
      if (zend_is_true(&retval)) {
1590
65
        zval_ptr_dtor(&retval);
1591
65
        return;
1592
65
      }
1593
0
      zval_ptr_dtor(&retval);
1594
0
    }
1595
0
    if (EG(exception)) {
1596
0
      return;
1597
0
    }
1598
0
    intern->inner.iterator->funcs->move_forward(intern->inner.iterator);
1599
0
  }
1600
25
  spl_dual_it_free(intern);
1601
25
}
1602
1603
static inline void spl_filter_it_rewind(zval *zthis, spl_dual_it_object *intern)
1604
70
{
1605
70
  spl_dual_it_rewind(intern);
1606
70
  spl_filter_it_fetch(zthis, intern);
1607
70
}
1608
1609
static inline void spl_filter_it_next(zval *zthis, spl_dual_it_object *intern)
1610
20
{
1611
20
  spl_dual_it_next(intern, 1);
1612
20
  spl_filter_it_fetch(zthis, intern);
1613
20
}
1614
1615
/* {{{ Rewind the iterator */
1616
PHP_METHOD(FilterIterator, rewind)
1617
70
{
1618
70
  spl_dual_it_object   *intern;
1619
1620
70
  ZEND_PARSE_PARAMETERS_NONE();
1621
1622
70
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1623
70
  spl_filter_it_rewind(ZEND_THIS, intern);
1624
70
} /* }}} */
1625
1626
/* {{{ Move the iterator forward */
1627
PHP_METHOD(FilterIterator, next)
1628
20
{
1629
20
  spl_dual_it_object   *intern;
1630
1631
20
  ZEND_PARSE_PARAMETERS_NONE();
1632
1633
20
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1634
20
  spl_filter_it_next(ZEND_THIS, intern);
1635
20
} /* }}} */
1636
1637
/* {{{ Create a RecursiveCallbackFilterIterator from a RecursiveIterator */
1638
PHP_METHOD(RecursiveCallbackFilterIterator, __construct)
1639
4
{
1640
4
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_RecursiveCallbackFilterIterator, spl_ce_RecursiveIterator, DIT_RecursiveCallbackFilterIterator);
1641
4
} /* }}} */
1642
1643
1644
/* {{{ Create a RecursiveFilterIterator from a RecursiveIterator */
1645
PHP_METHOD(RecursiveFilterIterator, __construct)
1646
0
{
1647
0
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_RecursiveFilterIterator, spl_ce_RecursiveIterator, DIT_RecursiveFilterIterator);
1648
0
} /* }}} */
1649
1650
/* {{{ Check whether the inner iterator's current element has children */
1651
PHP_METHOD(RecursiveFilterIterator, hasChildren)
1652
0
{
1653
0
  spl_dual_it_object   *intern;
1654
1655
0
  ZEND_PARSE_PARAMETERS_NONE();
1656
1657
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1658
1659
0
  zend_call_method_with_0_params(Z_OBJ(intern->inner.zobject), intern->inner.ce, NULL, "haschildren", return_value);
1660
0
} /* }}} */
1661
1662
/* {{{ Return the inner iterator's children contained in a RecursiveFilterIterator */
1663
PHP_METHOD(RecursiveFilterIterator, getChildren)
1664
0
{
1665
0
  spl_dual_it_object   *intern;
1666
1667
0
  ZEND_PARSE_PARAMETERS_NONE();
1668
1669
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1670
1671
0
  zval childrens;
1672
0
  zend_call_method_with_0_params(Z_OBJ(intern->inner.zobject), intern->inner.ce, NULL, "getchildren", &childrens);
1673
0
  if (Z_TYPE(childrens) == IS_UNDEF) {
1674
0
    RETURN_THROWS();
1675
0
  }
1676
1677
0
  zend_result is_initialized = object_init_with_constructor(return_value, Z_OBJCE_P(ZEND_THIS), 1, &childrens, NULL);
1678
0
  zval_ptr_dtor(&childrens);
1679
0
  if (is_initialized == FAILURE) {
1680
0
    RETURN_THROWS();
1681
0
  }
1682
0
} /* }}} */
1683
1684
/* {{{ Return the inner iterator's children contained in a RecursiveCallbackFilterIterator */
1685
PHP_METHOD(RecursiveCallbackFilterIterator, getChildren)
1686
0
{
1687
0
  spl_dual_it_object *intern;
1688
1689
0
  ZEND_PARSE_PARAMETERS_NONE();
1690
1691
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1692
1693
0
  zval params[2];
1694
0
  zend_call_method_with_0_params(Z_OBJ(intern->inner.zobject), intern->inner.ce, NULL, "getchildren", &params[0]);
1695
0
  if (Z_TYPE(params[0]) == IS_UNDEF) {
1696
0
    RETURN_THROWS();
1697
0
  }
1698
1699
  /* Get callable to pass to the constructor */
1700
0
  zend_get_callable_zval_from_fcc(&intern->u.callback_filter, &params[1]);
1701
1702
0
  zend_result is_initialized = object_init_with_constructor(return_value, Z_OBJCE_P(ZEND_THIS), 2, params, NULL);
1703
0
  zval_ptr_dtor(&params[0]);
1704
0
  zval_ptr_dtor(&params[1]);
1705
0
  if (is_initialized == FAILURE) {
1706
0
    RETURN_THROWS();
1707
0
  }
1708
0
} /* }}} */
1709
/* {{{ Create a ParentIterator from a RecursiveIterator */
1710
PHP_METHOD(ParentIterator, __construct)
1711
4
{
1712
4
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_ParentIterator, spl_ce_RecursiveIterator, DIT_ParentIterator);
1713
4
} /* }}} */
1714
1715
/* {{{ Create an RegexIterator from another iterator and a regular expression */
1716
PHP_METHOD(RegexIterator, __construct)
1717
4
{
1718
4
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_RegexIterator, zend_ce_iterator, DIT_RegexIterator);
1719
4
} /* }}} */
1720
1721
/* {{{ Calls the callback with the current value, the current key and the inner iterator as arguments */
1722
PHP_METHOD(CallbackFilterIterator, accept)
1723
65
{
1724
65
  spl_dual_it_object *intern;
1725
1726
65
  ZEND_PARSE_PARAMETERS_NONE();
1727
1728
65
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1729
1730
65
  if (Z_TYPE(intern->current.data) == IS_UNDEF || Z_TYPE(intern->current.key) == IS_UNDEF) {
1731
0
    RETURN_FALSE;
1732
0
  }
1733
1734
65
  zval params[3];
1735
65
  ZVAL_COPY_VALUE(&params[0], &intern->current.data);
1736
65
  ZVAL_COPY_VALUE(&params[1], &intern->current.key);
1737
65
  ZVAL_COPY_VALUE(&params[2], &intern->inner.zobject);
1738
1739
65
  zend_fcall_info_cache *fcc = &intern->u.callback_filter;
1740
1741
65
  zend_call_known_fcc(fcc, return_value, 3, params, NULL);
1742
65
  if (Z_ISUNDEF_P(return_value)) {
1743
0
    RETURN_FALSE;
1744
65
  } else if (Z_ISREF_P(return_value)) {
1745
12
    zend_unwrap_reference(return_value);
1746
12
  }
1747
65
}
1748
/* }}} */
1749
1750
/* {{{ Match (string)current() against regular expression */
1751
PHP_METHOD(RegexIterator, accept)
1752
0
{
1753
0
  spl_dual_it_object *intern;
1754
0
  zend_string *result, *subject;
1755
0
  size_t count = 0;
1756
0
  zval zcount, rv;
1757
0
  pcre2_match_data *match_data;
1758
0
  pcre2_code *re;
1759
0
  int rc;
1760
1761
0
  ZEND_PARSE_PARAMETERS_NONE();
1762
1763
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1764
1765
0
  if (Z_TYPE(intern->current.data) == IS_UNDEF) {
1766
0
    RETURN_FALSE;
1767
0
  }
1768
1769
0
  if (intern->u.regex.flags & REGIT_USE_KEY) {
1770
0
    subject = zval_get_string(&intern->current.key);
1771
0
  } else {
1772
0
    if (Z_TYPE(intern->current.data) == IS_ARRAY) {
1773
0
      RETURN_FALSE;
1774
0
    }
1775
0
    subject = zval_get_string(&intern->current.data);
1776
0
  }
1777
1778
  /* Exception during string conversion. */
1779
0
  if (EG(exception)) {
1780
0
    RETURN_THROWS();
1781
0
  }
1782
1783
0
  switch (intern->u.regex.mode)
1784
0
  {
1785
0
    case REGIT_MODE_MAX: /* won't happen but makes compiler happy */
1786
0
    case REGIT_MODE_MATCH:
1787
0
      re = php_pcre_pce_re(intern->u.regex.pce);
1788
0
      match_data = php_pcre_create_match_data(0, re);
1789
0
      if (!match_data) {
1790
0
        RETURN_FALSE;
1791
0
      }
1792
0
      rc = pcre2_match(re, (PCRE2_SPTR)ZSTR_VAL(subject), ZSTR_LEN(subject), 0, 0, match_data, php_pcre_mctx());
1793
0
      RETVAL_BOOL(rc >= 0);
1794
0
      php_pcre_free_match_data(match_data);
1795
0
      break;
1796
1797
0
    case REGIT_MODE_ALL_MATCHES:
1798
0
    case REGIT_MODE_GET_MATCH:
1799
0
      zval_ptr_dtor(&intern->current.data);
1800
0
      ZVAL_UNDEF(&intern->current.data);
1801
0
      php_pcre_match_impl(intern->u.regex.pce, subject, &zcount,
1802
0
        &intern->current.data, intern->u.regex.mode == REGIT_MODE_ALL_MATCHES, intern->u.regex.preg_flags, 0);
1803
0
      RETVAL_BOOL(Z_LVAL(zcount) > 0);
1804
0
      break;
1805
1806
0
    case REGIT_MODE_SPLIT:
1807
0
      zval_ptr_dtor(&intern->current.data);
1808
0
      ZVAL_UNDEF(&intern->current.data);
1809
0
      php_pcre_split_impl(intern->u.regex.pce, subject, &intern->current.data, -1, intern->u.regex.preg_flags);
1810
0
      count = zend_hash_num_elements(Z_ARRVAL(intern->current.data));
1811
0
      RETVAL_BOOL(count > 1);
1812
0
      break;
1813
1814
0
    case REGIT_MODE_REPLACE: {
1815
0
      zval *replacement = zend_read_property(intern->std.ce, Z_OBJ_P(ZEND_THIS), "replacement", sizeof("replacement")-1, 1, &rv);
1816
0
      zend_string *replacement_str = zval_try_get_string(replacement);
1817
1818
      /* Property type is ?string, so this should always succeed. */
1819
0
      ZEND_ASSERT(replacement_str != NULL);
1820
1821
0
      result = php_pcre_replace_impl(intern->u.regex.pce, subject, ZSTR_VAL(subject), ZSTR_LEN(subject), replacement_str, -1, &count);
1822
1823
0
      if (UNEXPECTED(!result)) {
1824
0
        zend_string_release(replacement_str);
1825
0
        zend_string_release_ex(subject, false);
1826
0
        RETURN_FALSE;
1827
0
      }
1828
1829
0
      if (intern->u.regex.flags & REGIT_USE_KEY) {
1830
0
        zval_ptr_dtor(&intern->current.key);
1831
0
        ZVAL_STR(&intern->current.key, result);
1832
0
      } else {
1833
0
        zval_ptr_dtor(&intern->current.data);
1834
0
        ZVAL_STR(&intern->current.data, result);
1835
0
      }
1836
1837
0
      zend_string_release(replacement_str);
1838
0
      RETVAL_BOOL(count > 0);
1839
0
    }
1840
0
  }
1841
1842
0
  if (intern->u.regex.flags & REGIT_INVERTED) {
1843
0
    RETVAL_BOOL(Z_TYPE_P(return_value) != IS_TRUE);
1844
0
  }
1845
0
  zend_string_release_ex(subject, false);
1846
0
} /* }}} */
1847
1848
/* {{{ Returns current regular expression */
1849
PHP_METHOD(RegexIterator, getRegex)
1850
0
{
1851
0
  spl_dual_it_object *intern;
1852
1853
0
  ZEND_PARSE_PARAMETERS_NONE();
1854
1855
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1856
1857
0
  RETURN_STR_COPY(intern->u.regex.regex);
1858
0
} /* }}} */
1859
1860
/* {{{ Returns current operation mode */
1861
PHP_METHOD(RegexIterator, getMode)
1862
0
{
1863
0
  spl_dual_it_object *intern;
1864
1865
0
  ZEND_PARSE_PARAMETERS_NONE();
1866
1867
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1868
1869
0
  RETURN_LONG(intern->u.regex.mode);
1870
0
} /* }}} */
1871
1872
/* {{{ Set new operation mode */
1873
PHP_METHOD(RegexIterator, setMode)
1874
0
{
1875
0
  spl_dual_it_object *intern;
1876
0
  zend_long mode;
1877
1878
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &mode) == FAILURE) {
1879
0
    RETURN_THROWS();
1880
0
  }
1881
1882
0
  if (mode < 0 || mode >= REGIT_MODE_MAX) {
1883
0
    zend_argument_value_error(1, "must be RegexIterator::MATCH, RegexIterator::GET_MATCH, "
1884
0
      "RegexIterator::ALL_MATCHES, RegexIterator::SPLIT, or RegexIterator::REPLACE");
1885
0
    RETURN_THROWS();
1886
0
  }
1887
1888
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1889
1890
0
  intern->u.regex.mode = mode;
1891
0
} /* }}} */
1892
1893
/* {{{ Returns current operation flags */
1894
PHP_METHOD(RegexIterator, getFlags)
1895
0
{
1896
0
  spl_dual_it_object *intern;
1897
1898
0
  ZEND_PARSE_PARAMETERS_NONE();
1899
1900
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1901
1902
0
  RETURN_LONG(intern->u.regex.flags);
1903
0
} /* }}} */
1904
1905
/* {{{ Set operation flags */
1906
PHP_METHOD(RegexIterator, setFlags)
1907
0
{
1908
0
  spl_dual_it_object *intern;
1909
0
  zend_long flags;
1910
1911
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &flags) == FAILURE) {
1912
0
    RETURN_THROWS();
1913
0
  }
1914
1915
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1916
1917
0
  intern->u.regex.flags = flags;
1918
0
} /* }}} */
1919
1920
/* {{{ Returns current PREG flags (if in use or NULL) */
1921
PHP_METHOD(RegexIterator, getPregFlags)
1922
0
{
1923
0
  spl_dual_it_object *intern;
1924
1925
0
  ZEND_PARSE_PARAMETERS_NONE();
1926
1927
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1928
1929
0
  RETURN_LONG(intern->u.regex.preg_flags);
1930
0
} /* }}} */
1931
1932
/* {{{ Set PREG flags */
1933
PHP_METHOD(RegexIterator, setPregFlags)
1934
0
{
1935
0
  spl_dual_it_object *intern;
1936
0
  zend_long preg_flags;
1937
1938
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &preg_flags) == FAILURE) {
1939
0
    RETURN_THROWS();
1940
0
  }
1941
1942
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1943
1944
0
  intern->u.regex.preg_flags = preg_flags;
1945
0
} /* }}} */
1946
1947
/* {{{ Create an RecursiveRegexIterator from another recursive iterator and a regular expression */
1948
PHP_METHOD(RecursiveRegexIterator, __construct)
1949
4
{
1950
4
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_RecursiveRegexIterator, spl_ce_RecursiveIterator, DIT_RecursiveRegexIterator);
1951
4
} /* }}} */
1952
1953
/* {{{ Return the inner iterator's children contained in a RecursiveRegexIterator */
1954
PHP_METHOD(RecursiveRegexIterator, getChildren)
1955
0
{
1956
0
  spl_dual_it_object   *intern;
1957
0
  zval                 retval;
1958
1959
0
  ZEND_PARSE_PARAMETERS_NONE();
1960
1961
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1962
1963
0
  zend_call_method_with_0_params(Z_OBJ(intern->inner.zobject), intern->inner.ce, NULL, "getchildren", &retval);
1964
0
  if (EG(exception)) {
1965
0
    zval_ptr_dtor(&retval);
1966
0
    RETURN_THROWS();
1967
0
  }
1968
1969
0
  zval args[5];
1970
0
  ZVAL_COPY_VALUE(&args[0], &retval);
1971
0
  ZVAL_STR_COPY(&args[1], intern->u.regex.regex);
1972
0
  ZVAL_LONG(&args[2], intern->u.regex.mode);
1973
0
  ZVAL_LONG(&args[3], intern->u.regex.flags);
1974
0
  ZVAL_LONG(&args[4], intern->u.regex.preg_flags);
1975
1976
0
  zend_result is_initialized = object_init_with_constructor(return_value, Z_OBJCE_P(ZEND_THIS), 5, args, NULL);
1977
1978
0
  zval_ptr_dtor(&args[0]);
1979
0
  zval_ptr_dtor_str(&args[1]);
1980
0
  if (is_initialized == FAILURE) {
1981
0
    RETURN_THROWS();
1982
0
  }
1983
0
} /* }}} */
1984
1985
PHP_METHOD(RecursiveRegexIterator, accept)
1986
0
{
1987
0
  spl_dual_it_object *intern;
1988
1989
0
  ZEND_PARSE_PARAMETERS_NONE();
1990
1991
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
1992
1993
0
  if (Z_TYPE(intern->current.data) == IS_UNDEF) {
1994
0
    RETURN_FALSE;
1995
0
  } else if (Z_TYPE(intern->current.data) == IS_ARRAY) {
1996
0
    RETURN_BOOL(zend_hash_num_elements(Z_ARRVAL(intern->current.data)) > 0);
1997
0
  }
1998
1999
0
  zend_call_method_with_0_params(Z_OBJ_P(ZEND_THIS), spl_ce_RegexIterator, NULL, "accept", return_value);
2000
0
}
2001
2002
/* {{{ spl_dual_it_free_storage */
2003
static void spl_dual_it_free_storage(zend_object *_object)
2004
496
{
2005
496
  spl_dual_it_object *object = spl_dual_it_from_obj(_object);
2006
2007
496
  spl_dual_it_free(object);
2008
2009
496
  if (object->inner.iterator) {
2010
110
    zend_iterator_dtor(object->inner.iterator);
2011
110
  }
2012
2013
496
  if (!Z_ISUNDEF(object->inner.zobject)) {
2014
110
    zval_ptr_dtor(&object->inner.zobject);
2015
110
  }
2016
2017
496
  if (object->dit_type == DIT_AppendIterator) {
2018
4
    zend_iterator_dtor(object->u.append.iterator);
2019
4
    if (Z_TYPE(object->u.append.zarrayit) != IS_UNDEF) {
2020
4
      zval_ptr_dtor(&object->u.append.zarrayit);
2021
4
    }
2022
4
  }
2023
2024
496
  if (object->dit_type == DIT_CachingIterator || object->dit_type == DIT_RecursiveCachingIterator) {
2025
0
    zval_ptr_dtor(&object->u.caching.zcache);
2026
0
  }
2027
2028
496
  if (object->dit_type == DIT_RegexIterator || object->dit_type == DIT_RecursiveRegexIterator) {
2029
0
    if (object->u.regex.pce) {
2030
0
      php_pcre_pce_decref(object->u.regex.pce);
2031
0
    }
2032
0
    if (object->u.regex.regex) {
2033
0
      zend_string_release_ex(object->u.regex.regex, 0);
2034
0
    }
2035
0
  }
2036
2037
496
  if (object->dit_type == DIT_CallbackFilterIterator || object->dit_type == DIT_RecursiveCallbackFilterIterator) {
2038
60
    if (ZEND_FCC_INITIALIZED(object->u.callback_filter)) {
2039
60
      zend_fcc_dtor(&object->u.callback_filter);
2040
60
    }
2041
60
  }
2042
2043
496
  zend_object_std_dtor(&object->std);
2044
496
}
2045
/* }}} */
2046
2047
static HashTable *spl_dual_it_get_gc(zend_object *obj, zval **table, int *n)
2048
931
{
2049
931
  spl_dual_it_object *object = spl_dual_it_from_obj(obj);
2050
931
  zend_get_gc_buffer *gc_buffer = zend_get_gc_buffer_create();
2051
2052
931
  if (object->inner.iterator) {
2053
298
    zend_get_gc_buffer_add_obj(gc_buffer, &object->inner.iterator->std);
2054
298
  }
2055
2056
931
  zend_get_gc_buffer_add_zval(gc_buffer, &object->current.data);
2057
931
  zend_get_gc_buffer_add_zval(gc_buffer, &object->current.key);
2058
931
  zend_get_gc_buffer_add_zval(gc_buffer, &object->inner.zobject);
2059
2060
931
  switch (object->dit_type) {
2061
633
    case DIT_Unknown:
2062
633
    case DIT_Default:
2063
709
    case DIT_IteratorIterator:
2064
709
    case DIT_NoRewindIterator:
2065
709
    case DIT_InfiniteIterator:
2066
711
    case DIT_LimitIterator:
2067
711
    case DIT_RegexIterator:
2068
711
    case DIT_RecursiveRegexIterator:
2069
      /* Nothing to do */
2070
711
      break;
2071
0
    case DIT_AppendIterator:
2072
0
      zend_get_gc_buffer_add_obj(gc_buffer, &object->u.append.iterator->std);
2073
0
      if (Z_TYPE(object->u.append.zarrayit) != IS_UNDEF) {
2074
0
        zend_get_gc_buffer_add_zval(gc_buffer, &object->u.append.zarrayit);
2075
0
      }
2076
0
      break;
2077
0
    case DIT_CachingIterator:
2078
0
    case DIT_RecursiveCachingIterator:
2079
0
      zend_get_gc_buffer_add_zval(gc_buffer, &object->u.caching.zcache);
2080
0
      zend_get_gc_buffer_add_zval(gc_buffer, &object->u.caching.zchildren);
2081
0
      break;
2082
220
    case DIT_CallbackFilterIterator:
2083
220
    case DIT_RecursiveCallbackFilterIterator:
2084
220
      if (ZEND_FCC_INITIALIZED(object->u.callback_filter)) {
2085
220
        zend_get_gc_buffer_add_fcc(gc_buffer, &object->u.callback_filter);
2086
220
      }
2087
220
      break;
2088
931
  }
2089
2090
931
  zend_get_gc_buffer_use(gc_buffer, table, n);
2091
931
  return zend_std_get_properties(obj);
2092
931
}
2093
2094
/* {{{ spl_dual_it_new */
2095
static zend_object *spl_dual_it_new(zend_class_entry *class_type)
2096
496
{
2097
496
  spl_dual_it_object *intern;
2098
2099
496
  intern = zend_object_alloc(sizeof(spl_dual_it_object), class_type);
2100
496
  intern->dit_type = DIT_Unknown;
2101
2102
496
  zend_object_std_init(&intern->std, class_type);
2103
496
  object_properties_init(&intern->std, class_type);
2104
2105
496
  return &intern->std;
2106
496
}
2107
/* }}} */
2108
2109
/* Returns the relative position for the current iterator position. */
2110
static zend_long spl_limit_it_relative_pos(spl_dual_it_object *intern)
2111
0
{
2112
0
  return intern->current.pos - intern->u.limit.offset;
2113
0
}
2114
2115
/* Returns the relative position for an arbitrary position. */
2116
static zend_long spl_limit_it_relative_pos_for(spl_dual_it_object *intern, zend_long pos)
2117
1
{
2118
1
  return pos - intern->u.limit.offset;
2119
1
}
2120
2121
static inline zend_result spl_limit_it_valid(spl_dual_it_object *intern)
2122
0
{
2123
  /* FAILURE / SUCCESS */
2124
0
  if (intern->u.limit.count != -1 &&
2125
0
    spl_limit_it_relative_pos(intern) >= intern->u.limit.count) {
2126
0
    return FAILURE;
2127
0
  }
2128
2129
0
  return spl_dual_it_valid(intern);
2130
0
}
2131
2132
static inline void spl_limit_it_seek(spl_dual_it_object *intern, zend_long pos)
2133
1
{
2134
1
  zval  zpos;
2135
2136
1
  spl_dual_it_free(intern);
2137
1
  if (pos < intern->u.limit.offset) {
2138
0
    zend_throw_exception_ex(spl_ce_OutOfBoundsException, 0, "Cannot seek to " ZEND_LONG_FMT " which is below the offset " ZEND_LONG_FMT, pos, intern->u.limit.offset);
2139
0
    return;
2140
0
  }
2141
1
  if (spl_limit_it_relative_pos_for(intern, pos) >= intern->u.limit.count && intern->u.limit.count != -1) {
2142
1
    zend_throw_exception_ex(spl_ce_OutOfBoundsException, 0, "Cannot seek to " ZEND_LONG_FMT " which is behind offset " ZEND_LONG_FMT " plus count " ZEND_LONG_FMT, pos, intern->u.limit.offset, intern->u.limit.count);
2143
1
    return;
2144
1
  }
2145
0
  if (pos != intern->current.pos && instanceof_function(intern->inner.ce, spl_ce_SeekableIterator)) {
2146
0
    ZVAL_LONG(&zpos, pos);
2147
0
    spl_dual_it_free(intern);
2148
0
    zend_call_method_with_1_params(Z_OBJ(intern->inner.zobject), intern->inner.ce, NULL, "seek", NULL, &zpos);
2149
0
    if (!EG(exception)) {
2150
0
      intern->current.pos = pos;
2151
0
      if (spl_limit_it_valid(intern) == SUCCESS) {
2152
0
        spl_dual_it_fetch(intern, 0);
2153
0
      }
2154
0
    }
2155
0
  } else {
2156
    /* emulate the forward seek, by next() calls */
2157
    /* a back ward seek is done by a previous rewind() */
2158
0
    if (pos < intern->current.pos) {
2159
0
      spl_dual_it_rewind(intern);
2160
0
    }
2161
0
    while (pos > intern->current.pos && spl_dual_it_valid(intern) == SUCCESS) {
2162
0
      spl_dual_it_next(intern, 1);
2163
0
    }
2164
0
    if (spl_dual_it_valid(intern) == SUCCESS) {
2165
0
      spl_dual_it_fetch(intern, 1);
2166
0
    }
2167
0
  }
2168
0
}
2169
2170
/* {{{ Construct a LimitIterator from an Iterator with a given starting offset and optionally a maximum count */
2171
PHP_METHOD(LimitIterator, __construct)
2172
5
{
2173
5
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_LimitIterator, zend_ce_iterator, DIT_LimitIterator);
2174
5
} /* }}} */
2175
2176
/* {{{ Rewind the iterator to the specified starting offset */
2177
PHP_METHOD(LimitIterator, rewind)
2178
1
{
2179
1
  spl_dual_it_object   *intern;
2180
2181
1
  ZEND_PARSE_PARAMETERS_NONE();
2182
2183
1
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2184
1
  spl_dual_it_rewind(intern);
2185
1
  spl_limit_it_seek(intern, intern->u.limit.offset);
2186
1
} /* }}} */
2187
2188
/* {{{ Check whether the current element is valid */
2189
PHP_METHOD(LimitIterator, valid)
2190
0
{
2191
0
  spl_dual_it_object   *intern;
2192
2193
0
  ZEND_PARSE_PARAMETERS_NONE();
2194
2195
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2196
2197
/*  RETURN_BOOL(spl_limit_it_valid(intern) == SUCCESS);*/
2198
0
  RETURN_BOOL((intern->u.limit.count == -1 || spl_limit_it_relative_pos(intern) < intern->u.limit.count) && Z_TYPE(intern->current.data) != IS_UNDEF);
2199
0
} /* }}} */
2200
2201
/* {{{ Move the iterator forward */
2202
PHP_METHOD(LimitIterator, next)
2203
0
{
2204
0
  spl_dual_it_object   *intern;
2205
2206
0
  ZEND_PARSE_PARAMETERS_NONE();
2207
2208
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2209
2210
0
  spl_dual_it_next(intern, 1);
2211
0
  if (intern->u.limit.count == -1 || spl_limit_it_relative_pos(intern) < intern->u.limit.count) {
2212
0
    spl_dual_it_fetch(intern, 1);
2213
0
  }
2214
0
} /* }}} */
2215
2216
/* {{{ Seek to the given position */
2217
PHP_METHOD(LimitIterator, seek)
2218
0
{
2219
0
  spl_dual_it_object   *intern;
2220
0
  zend_long                 pos;
2221
2222
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &pos) == FAILURE) {
2223
0
    RETURN_THROWS();
2224
0
  }
2225
2226
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2227
0
  spl_limit_it_seek(intern, pos);
2228
0
  RETURN_LONG(intern->current.pos);
2229
0
} /* }}} */
2230
2231
/* {{{ Return the current position */
2232
PHP_METHOD(LimitIterator, getPosition)
2233
0
{
2234
0
  spl_dual_it_object   *intern;
2235
2236
0
  ZEND_PARSE_PARAMETERS_NONE();
2237
2238
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2239
0
  RETURN_LONG(intern->current.pos);
2240
0
} /* }}} */
2241
2242
static inline int spl_caching_it_valid(spl_dual_it_object *intern)
2243
0
{
2244
0
  return intern->u.caching.flags & CIT_VALID ? SUCCESS : FAILURE;
2245
0
}
2246
2247
static inline int spl_caching_it_has_next(spl_dual_it_object *intern)
2248
0
{
2249
0
  return spl_dual_it_valid(intern);
2250
0
}
2251
2252
static inline void spl_caching_it_next(spl_dual_it_object *intern)
2253
0
{
2254
0
  if (spl_dual_it_fetch(intern, 1) == SUCCESS) {
2255
0
    intern->u.caching.flags |= CIT_VALID;
2256
    /* Full cache ? */
2257
0
    if (intern->u.caching.flags & CIT_FULL_CACHE) {
2258
0
      zval *key = &intern->current.key;
2259
0
      zval *data = &intern->current.data;
2260
2261
0
      ZVAL_DEREF(data);
2262
0
      array_set_zval_key(Z_ARRVAL(intern->u.caching.zcache), key, data);
2263
0
    }
2264
    /* Recursion ? */
2265
0
    if (intern->dit_type == DIT_RecursiveCachingIterator) {
2266
0
      zval retval;
2267
0
      zend_call_method_with_0_params(Z_OBJ(intern->inner.zobject), intern->inner.ce, NULL, "haschildren", &retval);
2268
0
      if (EG(exception)) {
2269
0
        zval_ptr_dtor(&retval);
2270
0
        if (intern->u.caching.flags & CIT_CATCH_GET_CHILD) {
2271
0
          zend_clear_exception();
2272
0
        } else {
2273
0
          return;
2274
0
        }
2275
0
      } else {
2276
0
        bool has_children = zend_is_true(&retval);
2277
0
        zval_ptr_dtor(&retval);
2278
2279
0
        if (has_children) {
2280
0
          zval args[2];
2281
2282
          /* Store the children in the first constructor argument */
2283
0
          zend_call_method_with_0_params(Z_OBJ(intern->inner.zobject), intern->inner.ce, NULL, "getchildren", &args[0]);
2284
0
          if (EG(exception)) {
2285
0
            zval_ptr_dtor(&args[0]);
2286
0
            if (intern->u.caching.flags & CIT_CATCH_GET_CHILD) {
2287
0
              zend_clear_exception();
2288
0
            } else {
2289
0
              return;
2290
0
            }
2291
0
          } else {
2292
0
            ZVAL_LONG(&args[1], intern->u.caching.flags & CIT_PUBLIC);
2293
2294
0
            zend_result is_initialized = object_init_with_constructor(
2295
0
              &intern->u.caching.zchildren,
2296
0
              spl_ce_RecursiveCachingIterator,
2297
0
              2,
2298
0
              args,
2299
0
              NULL
2300
0
            );
2301
0
            zval_ptr_dtor(&args[0]);
2302
0
            if (is_initialized == FAILURE) {
2303
0
              if (intern->u.caching.flags & CIT_CATCH_GET_CHILD) {
2304
0
                zend_clear_exception();
2305
0
              } else {
2306
0
                return;
2307
0
              }
2308
0
            }
2309
0
          }
2310
0
        }
2311
0
      }
2312
0
    }
2313
0
    if (intern->u.caching.flags & (CIT_TOSTRING_USE_INNER|CIT_CALL_TOSTRING)) {
2314
0
      if (intern->u.caching.flags & CIT_TOSTRING_USE_INNER) {
2315
0
        intern->u.caching.zstr = zval_get_string(&intern->inner.zobject);
2316
0
      } else {
2317
0
        intern->u.caching.zstr = zval_get_string(&intern->current.data);
2318
0
      }
2319
0
    }
2320
0
    spl_dual_it_next(intern, 0);
2321
0
  } else {
2322
0
    intern->u.caching.flags &= ~CIT_VALID;
2323
0
  }
2324
0
}
2325
2326
static inline void spl_caching_it_rewind(spl_dual_it_object *intern)
2327
0
{
2328
0
  spl_dual_it_rewind(intern);
2329
0
  zend_hash_clean(Z_ARRVAL(intern->u.caching.zcache));
2330
0
  spl_caching_it_next(intern);
2331
0
}
2332
2333
/* {{{ Construct a CachingIterator from an Iterator */
2334
PHP_METHOD(CachingIterator, __construct)
2335
4
{
2336
4
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_CachingIterator, zend_ce_iterator, DIT_CachingIterator);
2337
4
} /* }}} */
2338
2339
/* {{{ Rewind the iterator */
2340
PHP_METHOD(CachingIterator, rewind)
2341
0
{
2342
0
  spl_dual_it_object   *intern;
2343
2344
0
  ZEND_PARSE_PARAMETERS_NONE();
2345
2346
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2347
2348
0
  spl_caching_it_rewind(intern);
2349
0
} /* }}} */
2350
2351
/* {{{ Check whether the current element is valid */
2352
PHP_METHOD(CachingIterator, valid)
2353
0
{
2354
0
  spl_dual_it_object   *intern;
2355
2356
0
  ZEND_PARSE_PARAMETERS_NONE();
2357
2358
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2359
2360
0
  RETURN_BOOL(spl_caching_it_valid(intern) == SUCCESS);
2361
0
} /* }}} */
2362
2363
/* {{{ Move the iterator forward */
2364
PHP_METHOD(CachingIterator, next)
2365
0
{
2366
0
  spl_dual_it_object   *intern;
2367
2368
0
  ZEND_PARSE_PARAMETERS_NONE();
2369
2370
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2371
2372
0
  spl_caching_it_next(intern);
2373
0
} /* }}} */
2374
2375
/* {{{ Check whether the inner iterator has a valid next element */
2376
PHP_METHOD(CachingIterator, hasNext)
2377
0
{
2378
0
  spl_dual_it_object   *intern;
2379
2380
0
  ZEND_PARSE_PARAMETERS_NONE();
2381
2382
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2383
2384
0
  RETURN_BOOL(spl_caching_it_has_next(intern) == SUCCESS);
2385
0
} /* }}} */
2386
2387
/* {{{ Return the string representation of the current element */
2388
PHP_METHOD(CachingIterator, __toString)
2389
0
{
2390
0
  spl_dual_it_object *intern;
2391
2392
0
  ZEND_PARSE_PARAMETERS_NONE();
2393
2394
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2395
2396
0
  if (!(intern->u.caching.flags & (CIT_CALL_TOSTRING|CIT_TOSTRING_USE_KEY|CIT_TOSTRING_USE_CURRENT|CIT_TOSTRING_USE_INNER))) {
2397
0
    zend_throw_exception_ex(spl_ce_BadMethodCallException, 0, "%s does not fetch string value (see CachingIterator::__construct)", ZSTR_VAL(Z_OBJCE_P(ZEND_THIS)->name));
2398
0
    RETURN_THROWS();
2399
0
  }
2400
2401
0
  if (intern->u.caching.flags & CIT_TOSTRING_USE_KEY) {
2402
0
    ZVAL_COPY(return_value, &intern->current.key);
2403
0
    convert_to_string(return_value);
2404
0
    return;
2405
0
  } else if (intern->u.caching.flags & CIT_TOSTRING_USE_CURRENT) {
2406
0
    ZVAL_COPY(return_value, &intern->current.data);
2407
0
    convert_to_string(return_value);
2408
0
    return;
2409
0
  }
2410
0
  if (intern->u.caching.zstr) {
2411
0
    RETURN_STR_COPY(intern->u.caching.zstr);
2412
0
  } else {
2413
0
    RETURN_EMPTY_STRING();
2414
0
  }
2415
0
} /* }}} */
2416
2417
/* {{{ Set given index in cache */
2418
PHP_METHOD(CachingIterator, offsetSet)
2419
0
{
2420
0
  spl_dual_it_object   *intern;
2421
0
  zend_string *key;
2422
0
  zval *value;
2423
2424
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "Sz", &key, &value) == FAILURE) {
2425
0
    RETURN_THROWS();
2426
0
  }
2427
2428
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2429
2430
0
  if (!(intern->u.caching.flags & CIT_FULL_CACHE)) {
2431
0
    zend_throw_exception_ex(spl_ce_BadMethodCallException, 0, "%s does not use a full cache (see CachingIterator::__construct)", ZSTR_VAL(Z_OBJCE_P(ZEND_THIS)->name));
2432
0
    RETURN_THROWS();
2433
0
  }
2434
2435
0
  Z_TRY_ADDREF_P(value);
2436
0
  zend_symtable_update(Z_ARRVAL(intern->u.caching.zcache), key, value);
2437
0
}
2438
/* }}} */
2439
2440
/* {{{ Return the internal cache if used */
2441
PHP_METHOD(CachingIterator, offsetGet)
2442
0
{
2443
0
  spl_dual_it_object   *intern;
2444
0
  zend_string *key;
2445
0
  zval *value;
2446
2447
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "S", &key) == FAILURE) {
2448
0
    RETURN_THROWS();
2449
0
  }
2450
2451
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2452
2453
0
  if (!(intern->u.caching.flags & CIT_FULL_CACHE)) {
2454
0
    zend_throw_exception_ex(spl_ce_BadMethodCallException, 0, "%s does not use a full cache (see CachingIterator::__construct)", ZSTR_VAL(Z_OBJCE_P(ZEND_THIS)->name));
2455
0
    RETURN_THROWS();
2456
0
  }
2457
2458
0
  if ((value = zend_symtable_find(Z_ARRVAL(intern->u.caching.zcache), key)) == NULL) {
2459
0
    zend_error(E_WARNING, "Undefined array key \"%s\"", ZSTR_VAL(key));
2460
0
    return;
2461
0
  }
2462
2463
0
  RETURN_COPY_DEREF(value);
2464
0
}
2465
/* }}} */
2466
2467
/* {{{ Unset given index in cache */
2468
PHP_METHOD(CachingIterator, offsetUnset)
2469
0
{
2470
0
  spl_dual_it_object   *intern;
2471
0
  zend_string *key;
2472
2473
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2474
2475
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "S", &key) == FAILURE) {
2476
0
    RETURN_THROWS();
2477
0
  }
2478
2479
0
  if (!(intern->u.caching.flags & CIT_FULL_CACHE)) {
2480
0
    zend_throw_exception_ex(spl_ce_BadMethodCallException, 0, "%s does not use a full cache (see CachingIterator::__construct)", ZSTR_VAL(Z_OBJCE_P(ZEND_THIS)->name));
2481
0
    RETURN_THROWS();
2482
0
  }
2483
2484
0
  zend_symtable_del(Z_ARRVAL(intern->u.caching.zcache), key);
2485
0
}
2486
/* }}} */
2487
2488
/* {{{ Return whether the requested index exists */
2489
PHP_METHOD(CachingIterator, offsetExists)
2490
0
{
2491
0
  spl_dual_it_object   *intern;
2492
0
  zend_string *key;
2493
2494
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "S", &key) == FAILURE) {
2495
0
    RETURN_THROWS();
2496
0
  }
2497
2498
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2499
2500
0
  if (!(intern->u.caching.flags & CIT_FULL_CACHE)) {
2501
0
    zend_throw_exception_ex(spl_ce_BadMethodCallException, 0, "%s does not use a full cache (see CachingIterator::__construct)", ZSTR_VAL(Z_OBJCE_P(ZEND_THIS)->name));
2502
0
    RETURN_THROWS();
2503
0
  }
2504
2505
0
  RETURN_BOOL(zend_symtable_exists(Z_ARRVAL(intern->u.caching.zcache), key));
2506
0
}
2507
/* }}} */
2508
2509
/* {{{ Return the cache */
2510
PHP_METHOD(CachingIterator, getCache)
2511
0
{
2512
0
  spl_dual_it_object *intern;
2513
2514
0
  ZEND_PARSE_PARAMETERS_NONE();
2515
2516
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2517
2518
0
  if (!(intern->u.caching.flags & CIT_FULL_CACHE)) {
2519
0
    zend_throw_exception_ex(spl_ce_BadMethodCallException, 0, "%s does not use a full cache (see CachingIterator::__construct)", ZSTR_VAL(Z_OBJCE_P(ZEND_THIS)->name));
2520
0
    RETURN_THROWS();
2521
0
  }
2522
2523
0
  ZVAL_COPY(return_value, &intern->u.caching.zcache);
2524
0
}
2525
/* }}} */
2526
2527
/* {{{ Return the internal flags */
2528
PHP_METHOD(CachingIterator, getFlags)
2529
0
{
2530
0
  spl_dual_it_object   *intern;
2531
2532
0
  ZEND_PARSE_PARAMETERS_NONE();
2533
2534
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2535
2536
0
  RETURN_LONG(intern->u.caching.flags);
2537
0
}
2538
/* }}} */
2539
2540
/* {{{ Set the internal flags */
2541
PHP_METHOD(CachingIterator, setFlags)
2542
0
{
2543
0
  spl_dual_it_object   *intern;
2544
0
  zend_long flags;
2545
2546
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &flags) == FAILURE) {
2547
0
    RETURN_THROWS();
2548
0
  }
2549
2550
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2551
2552
0
  if (spl_cit_check_flags(flags) != SUCCESS) {
2553
0
    zend_argument_value_error(1, "must contain only one of CachingIterator::CALL_TOSTRING, "
2554
0
      "CachingIterator::TOSTRING_USE_KEY, CachingIterator::TOSTRING_USE_CURRENT, "
2555
0
      "or CachingIterator::TOSTRING_USE_INNER");
2556
0
    RETURN_THROWS();
2557
0
  }
2558
0
  if ((intern->u.caching.flags & CIT_CALL_TOSTRING) != 0 && (flags & CIT_CALL_TOSTRING) == 0) {
2559
0
    zend_throw_exception(spl_ce_InvalidArgumentException, "Unsetting flag CALL_TO_STRING is not possible", 0);
2560
0
    RETURN_THROWS();
2561
0
  }
2562
0
  if ((intern->u.caching.flags & CIT_TOSTRING_USE_INNER) != 0 && (flags & CIT_TOSTRING_USE_INNER) == 0) {
2563
0
    zend_throw_exception(spl_ce_InvalidArgumentException, "Unsetting flag TOSTRING_USE_INNER is not possible", 0);
2564
0
    RETURN_THROWS();
2565
0
  }
2566
0
  if ((flags & CIT_FULL_CACHE) != 0 && (intern->u.caching.flags & CIT_FULL_CACHE) == 0) {
2567
    /* clear on (re)enable */
2568
0
    zend_hash_clean(Z_ARRVAL(intern->u.caching.zcache));
2569
0
  }
2570
0
  intern->u.caching.flags = (intern->u.caching.flags & ~CIT_PUBLIC) | (flags & CIT_PUBLIC);
2571
0
}
2572
/* }}} */
2573
2574
/* {{{ Number of cached elements */
2575
PHP_METHOD(CachingIterator, count)
2576
0
{
2577
0
  spl_dual_it_object   *intern;
2578
2579
0
  ZEND_PARSE_PARAMETERS_NONE();
2580
2581
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2582
2583
0
  if (!(intern->u.caching.flags & CIT_FULL_CACHE)) {
2584
0
    zend_throw_exception_ex(spl_ce_BadMethodCallException, 0, "%s does not use a full cache (see CachingIterator::__construct)", ZSTR_VAL(Z_OBJCE_P(ZEND_THIS)->name));
2585
0
    RETURN_THROWS();
2586
0
  }
2587
2588
0
  RETURN_LONG(zend_hash_num_elements(Z_ARRVAL(intern->u.caching.zcache)));
2589
0
}
2590
/* }}} */
2591
2592
/* {{{ Create an iterator from a RecursiveIterator */
2593
PHP_METHOD(RecursiveCachingIterator, __construct)
2594
4
{
2595
4
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_RecursiveCachingIterator, spl_ce_RecursiveIterator, DIT_RecursiveCachingIterator);
2596
4
} /* }}} */
2597
2598
/* {{{ Check whether the current element of the inner iterator has children */
2599
PHP_METHOD(RecursiveCachingIterator, hasChildren)
2600
0
{
2601
0
  spl_dual_it_object   *intern;
2602
2603
0
  ZEND_PARSE_PARAMETERS_NONE();
2604
2605
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2606
2607
0
  RETURN_BOOL(Z_TYPE(intern->u.caching.zchildren) != IS_UNDEF);
2608
0
} /* }}} */
2609
2610
/* {{{ Return the inner iterator's children as a RecursiveCachingIterator */
2611
PHP_METHOD(RecursiveCachingIterator, getChildren)
2612
0
{
2613
0
  spl_dual_it_object   *intern;
2614
2615
0
  ZEND_PARSE_PARAMETERS_NONE();
2616
2617
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2618
2619
0
  if (Z_TYPE(intern->u.caching.zchildren) != IS_UNDEF) {
2620
0
    zval *value = &intern->u.caching.zchildren;
2621
2622
0
    RETURN_COPY_DEREF(value);
2623
0
  } else {
2624
0
    RETURN_NULL();
2625
0
  }
2626
0
} /* }}} */
2627
2628
/* {{{ Create an iterator from anything that is traversable */
2629
PHP_METHOD(IteratorIterator, __construct)
2630
53
{
2631
53
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_IteratorIterator, zend_ce_traversable, DIT_IteratorIterator);
2632
53
} /* }}} */
2633
2634
/* {{{ Create an iterator from another iterator */
2635
PHP_METHOD(NoRewindIterator, __construct)
2636
4
{
2637
4
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_NoRewindIterator, zend_ce_iterator, DIT_NoRewindIterator);
2638
4
} /* }}} */
2639
2640
/* {{{ Prevent a call to inner iterators rewind() */
2641
PHP_METHOD(NoRewindIterator, rewind)
2642
0
{
2643
0
  ZEND_PARSE_PARAMETERS_NONE();
2644
  /* nothing to do */
2645
0
} /* }}} */
2646
2647
/* {{{ Return inner iterators valid() */
2648
PHP_METHOD(NoRewindIterator, valid)
2649
0
{
2650
0
  spl_dual_it_object   *intern;
2651
2652
0
  ZEND_PARSE_PARAMETERS_NONE();
2653
2654
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2655
0
  RETURN_BOOL(intern->inner.iterator->funcs->valid(intern->inner.iterator) == SUCCESS);
2656
0
} /* }}} */
2657
2658
/* {{{ Return inner iterators key() */
2659
PHP_METHOD(NoRewindIterator, key)
2660
0
{
2661
0
  spl_dual_it_object   *intern;
2662
2663
0
  ZEND_PARSE_PARAMETERS_NONE();
2664
2665
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2666
2667
0
  if (intern->inner.iterator->funcs->get_current_key) {
2668
0
    intern->inner.iterator->funcs->get_current_key(intern->inner.iterator, return_value);
2669
0
  } else {
2670
0
    RETURN_NULL();
2671
0
  }
2672
0
} /* }}} */
2673
2674
/* {{{ Return inner iterators current() */
2675
PHP_METHOD(NoRewindIterator, current)
2676
0
{
2677
0
  spl_dual_it_object   *intern;
2678
0
  zval *data;
2679
2680
0
  ZEND_PARSE_PARAMETERS_NONE();
2681
2682
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2683
0
  data = intern->inner.iterator->funcs->get_current_data(intern->inner.iterator);
2684
0
  if (data) {
2685
0
    RETURN_COPY_DEREF(data);
2686
0
  }
2687
0
} /* }}} */
2688
2689
/* {{{ Return inner iterators next() */
2690
PHP_METHOD(NoRewindIterator, next)
2691
0
{
2692
0
  spl_dual_it_object   *intern;
2693
2694
0
  ZEND_PARSE_PARAMETERS_NONE();
2695
2696
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2697
0
  intern->inner.iterator->funcs->move_forward(intern->inner.iterator);
2698
0
} /* }}} */
2699
2700
/* {{{ Create an iterator from another iterator */
2701
PHP_METHOD(InfiniteIterator, __construct)
2702
0
{
2703
0
  spl_dual_it_construct(INTERNAL_FUNCTION_PARAM_PASSTHRU, spl_ce_InfiniteIterator, zend_ce_iterator, DIT_InfiniteIterator);
2704
0
} /* }}} */
2705
2706
/* {{{ Prevent a call to inner iterators rewind() (internally the current data will be fetched if valid()) */
2707
PHP_METHOD(InfiniteIterator, next)
2708
0
{
2709
0
  spl_dual_it_object   *intern;
2710
2711
0
  ZEND_PARSE_PARAMETERS_NONE();
2712
2713
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2714
2715
0
  spl_dual_it_next(intern, 1);
2716
0
  if (spl_dual_it_valid(intern) == SUCCESS) {
2717
0
    spl_dual_it_fetch(intern, 0);
2718
0
  } else {
2719
0
    spl_dual_it_rewind(intern);
2720
0
    if (spl_dual_it_valid(intern) == SUCCESS) {
2721
0
      spl_dual_it_fetch(intern, 0);
2722
0
    }
2723
0
  }
2724
0
} /* }}} */
2725
2726
/* {{{ Does nothing  */
2727
PHP_METHOD(EmptyIterator, rewind)
2728
0
{
2729
0
  ZEND_PARSE_PARAMETERS_NONE();
2730
0
} /* }}} */
2731
2732
/* {{{ Return false */
2733
PHP_METHOD(EmptyIterator, valid)
2734
0
{
2735
0
  ZEND_PARSE_PARAMETERS_NONE();
2736
0
  RETURN_FALSE;
2737
0
} /* }}} */
2738
2739
/* {{{ Throws exception BadMethodCallException */
2740
PHP_METHOD(EmptyIterator, key)
2741
0
{
2742
0
  ZEND_PARSE_PARAMETERS_NONE();
2743
0
  zend_throw_exception(spl_ce_BadMethodCallException, "Accessing the key of an EmptyIterator", 0);
2744
0
} /* }}} */
2745
2746
/* {{{ Throws exception BadMethodCallException */
2747
PHP_METHOD(EmptyIterator, current)
2748
0
{
2749
0
  ZEND_PARSE_PARAMETERS_NONE();
2750
0
  zend_throw_exception(spl_ce_BadMethodCallException, "Accessing the value of an EmptyIterator", 0);
2751
0
} /* }}} */
2752
2753
/* {{{ Does nothing */
2754
PHP_METHOD(EmptyIterator, next)
2755
0
{
2756
0
  ZEND_PARSE_PARAMETERS_NONE();
2757
0
} /* }}} */
2758
2759
static zend_result spl_append_it_next_iterator(spl_dual_it_object *intern) /* {{{*/
2760
0
{
2761
0
  spl_dual_it_free(intern);
2762
2763
0
  if (!Z_ISUNDEF(intern->inner.zobject)) {
2764
0
    zval_ptr_dtor(&intern->inner.zobject);
2765
0
    ZVAL_UNDEF(&intern->inner.zobject);
2766
0
    intern->inner.ce = NULL;
2767
0
    if (intern->inner.iterator) {
2768
0
      zend_iterator_dtor(intern->inner.iterator);
2769
0
      intern->inner.iterator = NULL;
2770
0
    }
2771
0
  }
2772
0
  if (intern->u.append.iterator->funcs->valid(intern->u.append.iterator) == SUCCESS) {
2773
0
    zval *it;
2774
2775
0
    it  = intern->u.append.iterator->funcs->get_current_data(intern->u.append.iterator);
2776
0
    ZVAL_COPY(&intern->inner.zobject, it);
2777
0
    intern->inner.ce = Z_OBJCE_P(it);
2778
0
    intern->inner.iterator = intern->inner.ce->get_iterator(intern->inner.ce, it, 0);
2779
0
    spl_dual_it_rewind(intern);
2780
0
    return SUCCESS;
2781
0
  } else {
2782
0
    return FAILURE;
2783
0
  }
2784
0
} /* }}} */
2785
2786
static void spl_append_it_fetch(spl_dual_it_object *intern) /* {{{*/
2787
0
{
2788
0
  while (spl_dual_it_valid(intern) != SUCCESS) {
2789
0
    intern->u.append.iterator->funcs->move_forward(intern->u.append.iterator);
2790
0
    if (spl_append_it_next_iterator(intern) != SUCCESS) {
2791
0
      return;
2792
0
    }
2793
0
  }
2794
0
  spl_dual_it_fetch(intern, 0);
2795
0
} /* }}} */
2796
2797
static void spl_append_it_next(spl_dual_it_object *intern) /* {{{ */
2798
0
{
2799
0
  if (spl_dual_it_valid(intern) == SUCCESS) {
2800
0
    spl_dual_it_next(intern, 1);
2801
0
  }
2802
0
  spl_append_it_fetch(intern);
2803
0
} /* }}} */
2804
2805
/* {{{ Create an AppendIterator */
2806
PHP_METHOD(AppendIterator, __construct)
2807
4
{
2808
4
  ZEND_PARSE_PARAMETERS_NONE();
2809
2810
4
  spl_dual_it_object *intern = Z_SPLDUAL_IT_P(ZEND_THIS);
2811
2812
  /* TODO: This should be converted to a normal Error as this is triggered when calling the constructor twice */
2813
4
  if (intern->dit_type != DIT_Unknown) {
2814
0
    zend_throw_exception_ex(spl_ce_BadMethodCallException, 0, "%s::getIterator() must be called exactly once per instance", ZSTR_VAL(spl_ce_AppendIterator->name));
2815
0
    RETURN_THROWS();
2816
0
  }
2817
2818
4
  intern->dit_type = DIT_AppendIterator;
2819
4
  object_init_ex(&intern->u.append.zarrayit, spl_ce_ArrayIterator);
2820
4
  zend_call_method_with_0_params(Z_OBJ(intern->u.append.zarrayit), spl_ce_ArrayIterator, &spl_ce_ArrayIterator->constructor, "__construct", NULL);
2821
4
  intern->u.append.iterator = spl_ce_ArrayIterator->get_iterator(spl_ce_ArrayIterator, &intern->u.append.zarrayit, 0);
2822
2823
4
} /* }}} */
2824
2825
/* {{{ Append an iterator */
2826
PHP_METHOD(AppendIterator, append)
2827
0
{
2828
0
  spl_dual_it_object   *intern;
2829
0
  zval *it;
2830
2831
0
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "O", &it, zend_ce_iterator) == FAILURE) {
2832
0
    RETURN_THROWS();
2833
0
  }
2834
2835
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2836
2837
0
  if (intern->u.append.iterator->funcs->valid(intern->u.append.iterator) == SUCCESS && spl_dual_it_valid(intern) != SUCCESS) {
2838
0
    spl_array_iterator_append(&intern->u.append.zarrayit, it);
2839
0
    intern->u.append.iterator->funcs->move_forward(intern->u.append.iterator);
2840
0
  }else{
2841
0
    spl_array_iterator_append(&intern->u.append.zarrayit, it);
2842
0
  }
2843
2844
0
  if (!intern->inner.iterator || spl_dual_it_valid(intern) != SUCCESS) {
2845
0
    if (intern->u.append.iterator->funcs->valid(intern->u.append.iterator) != SUCCESS) {
2846
0
      intern->u.append.iterator->funcs->rewind(intern->u.append.iterator);
2847
0
    }
2848
0
    do {
2849
0
      spl_append_it_next_iterator(intern);
2850
0
    } while (Z_OBJ(intern->inner.zobject) != Z_OBJ_P(it));
2851
0
    spl_append_it_fetch(intern);
2852
0
  }
2853
0
} /* }}} */
2854
2855
/* {{{ Get the current element value */
2856
PHP_METHOD(AppendIterator, current)
2857
0
{
2858
0
  spl_dual_it_object   *intern;
2859
2860
0
  ZEND_PARSE_PARAMETERS_NONE();
2861
2862
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2863
2864
0
  spl_dual_it_fetch(intern, 1);
2865
0
  if (Z_TYPE(intern->current.data) != IS_UNDEF) {
2866
0
    RETURN_COPY_DEREF(&intern->current.data);
2867
0
  } else {
2868
0
    RETURN_NULL();
2869
0
  }
2870
0
} /* }}} */
2871
2872
/* {{{ Rewind to the first iterator and rewind the first iterator, too */
2873
PHP_METHOD(AppendIterator, rewind)
2874
0
{
2875
0
  spl_dual_it_object   *intern;
2876
2877
0
  ZEND_PARSE_PARAMETERS_NONE();
2878
2879
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2880
2881
0
  intern->u.append.iterator->funcs->rewind(intern->u.append.iterator);
2882
0
  if (spl_append_it_next_iterator(intern) == SUCCESS) {
2883
0
    spl_append_it_fetch(intern);
2884
0
  }
2885
0
} /* }}} */
2886
2887
/* {{{ Check if the current state is valid */
2888
PHP_METHOD(AppendIterator, valid)
2889
0
{
2890
0
  spl_dual_it_object   *intern;
2891
2892
0
  ZEND_PARSE_PARAMETERS_NONE();
2893
2894
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2895
2896
0
  RETURN_BOOL(Z_TYPE(intern->current.data) != IS_UNDEF);
2897
0
} /* }}} */
2898
2899
/* {{{ Forward to next element */
2900
PHP_METHOD(AppendIterator, next)
2901
0
{
2902
0
  spl_dual_it_object   *intern;
2903
2904
0
  ZEND_PARSE_PARAMETERS_NONE();
2905
2906
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2907
2908
0
  spl_append_it_next(intern);
2909
0
} /* }}} */
2910
2911
/* {{{ Get index of iterator */
2912
PHP_METHOD(AppendIterator, getIteratorIndex)
2913
0
{
2914
0
  spl_dual_it_object   *intern;
2915
2916
0
  ZEND_PARSE_PARAMETERS_NONE();
2917
2918
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2919
2920
0
  APPENDIT_CHECK_CTOR(intern);
2921
0
  spl_array_iterator_key(&intern->u.append.zarrayit, return_value);
2922
0
} /* }}} */
2923
2924
/* {{{ Get access to inner ArrayIterator */
2925
PHP_METHOD(AppendIterator, getArrayIterator)
2926
0
{
2927
0
  spl_dual_it_object   *intern;
2928
0
  zval *value;
2929
2930
0
  ZEND_PARSE_PARAMETERS_NONE();
2931
2932
0
  SPL_FETCH_AND_CHECK_DUAL_IT(intern, ZEND_THIS);
2933
2934
0
  value = &intern->u.append.zarrayit;
2935
0
  RETURN_COPY_DEREF(value);
2936
0
} /* }}} */
2937
2938
PHPAPI zend_result spl_iterator_apply(zval *obj, spl_iterator_apply_func_t apply_func, void *puser)
2939
55
{
2940
55
  zend_object_iterator   *iter;
2941
55
  zend_class_entry       *ce = Z_OBJCE_P(obj);
2942
2943
55
  iter = ce->get_iterator(ce, obj, 0);
2944
2945
55
  if (EG(exception)) {
2946
0
    goto done;
2947
0
  }
2948
2949
55
  iter->index = 0;
2950
55
  if (iter->funcs->rewind) {
2951
55
    iter->funcs->rewind(iter);
2952
55
    if (EG(exception)) {
2953
4
      goto done;
2954
4
    }
2955
55
  }
2956
2957
149
  while (iter->funcs->valid(iter) == SUCCESS) {
2958
100
    if (EG(exception)) {
2959
0
      goto done;
2960
0
    }
2961
100
    if (apply_func(iter, puser) == ZEND_HASH_APPLY_STOP || EG(exception)) {
2962
2
      goto done;
2963
2
    }
2964
98
    iter->index++;
2965
98
    iter->funcs->move_forward(iter);
2966
98
    if (EG(exception)) {
2967
0
      goto done;
2968
0
    }
2969
98
  }
2970
2971
55
done:
2972
55
  if (iter) {
2973
55
    zend_iterator_dtor(iter);
2974
55
  }
2975
55
  return EG(exception) ? FAILURE : SUCCESS;
2976
51
}
2977
/* }}} */
2978
2979
static int spl_iterator_to_array_apply(zend_object_iterator *iter, void *puser) /* {{{ */
2980
94
{
2981
94
  zval *data, *return_value = (zval*)puser;
2982
2983
94
  data = iter->funcs->get_current_data(iter);
2984
94
  if (EG(exception)) {
2985
2
    return ZEND_HASH_APPLY_STOP;
2986
2
  }
2987
92
  if (data == NULL) {
2988
0
    return ZEND_HASH_APPLY_STOP;
2989
0
  }
2990
92
  if (iter->funcs->get_current_key) {
2991
92
    zval key;
2992
92
    iter->funcs->get_current_key(iter, &key);
2993
92
    if (EG(exception)) {
2994
0
      return ZEND_HASH_APPLY_STOP;
2995
0
    }
2996
92
    array_set_zval_key(Z_ARRVAL_P(return_value), &key, data);
2997
92
    zval_ptr_dtor(&key);
2998
92
  } else {
2999
0
    Z_TRY_ADDREF_P(data);
3000
0
    add_next_index_zval(return_value, data);
3001
0
  }
3002
92
  return ZEND_HASH_APPLY_KEEP;
3003
92
}
3004
/* }}} */
3005
3006
static int spl_iterator_to_values_apply(zend_object_iterator *iter, void *puser) /* {{{ */
3007
6
{
3008
6
  zval *data, *return_value = (zval*)puser;
3009
3010
6
  data = iter->funcs->get_current_data(iter);
3011
6
  if (EG(exception)) {
3012
0
    return ZEND_HASH_APPLY_STOP;
3013
0
  }
3014
6
  if (data == NULL) {
3015
0
    return ZEND_HASH_APPLY_STOP;
3016
0
  }
3017
6
  Z_TRY_ADDREF_P(data);
3018
6
  add_next_index_zval(return_value, data);
3019
6
  return ZEND_HASH_APPLY_KEEP;
3020
6
}
3021
/* }}} */
3022
3023
/* {{{ Copy the iterator into an array */
3024
PHP_FUNCTION(iterator_to_array)
3025
61
{
3026
61
  zval  *obj;
3027
61
  bool use_keys = 1;
3028
3029
183
  ZEND_PARSE_PARAMETERS_START(1, 2)
3030
244
    Z_PARAM_ITERABLE(obj)
3031
55
    Z_PARAM_OPTIONAL
3032
122
    Z_PARAM_BOOL(use_keys)
3033
61
  ZEND_PARSE_PARAMETERS_END();
3034
3035
55
  if (Z_TYPE_P(obj) == IS_ARRAY) {
3036
0
    if (use_keys) {
3037
0
      RETURN_COPY(obj);
3038
0
    } else {
3039
0
      RETURN_ARR(zend_array_to_list(Z_ARRVAL_P(obj)));
3040
0
    }
3041
0
  }
3042
3043
55
  array_init(return_value);
3044
55
  spl_iterator_apply(obj, use_keys ? spl_iterator_to_array_apply : spl_iterator_to_values_apply, (void*)return_value);
3045
55
} /* }}} */
3046
3047
static int spl_iterator_count_apply(zend_object_iterator *iter, void *puser) /* {{{ */
3048
0
{
3049
0
  if (UNEXPECTED(*(zend_long*)puser == ZEND_LONG_MAX)) {
3050
0
    return ZEND_HASH_APPLY_STOP;
3051
0
  }
3052
0
  (*(zend_long*)puser)++;
3053
0
  return ZEND_HASH_APPLY_KEEP;
3054
0
}
3055
/* }}} */
3056
3057
/* {{{ Count the elements in an iterator */
3058
PHP_FUNCTION(iterator_count)
3059
0
{
3060
0
  zval  *obj;
3061
0
  zend_long  count = 0;
3062
3063
0
  ZEND_PARSE_PARAMETERS_START(1, 1)
3064
0
    Z_PARAM_ITERABLE(obj)
3065
0
  ZEND_PARSE_PARAMETERS_END();
3066
3067
0
  if (Z_TYPE_P(obj) == IS_ARRAY) {
3068
0
    count =  zend_hash_num_elements(Z_ARRVAL_P(obj));
3069
0
  } else {
3070
0
    if (spl_iterator_apply(obj, spl_iterator_count_apply, (void*)&count) == FAILURE) {
3071
0
      RETURN_THROWS();
3072
0
    }
3073
0
  }
3074
3075
0
  RETURN_LONG(count);
3076
0
}
3077
/* }}} */
3078
3079
typedef struct {
3080
  zend_long count;
3081
  HashTable *params_ht;
3082
  zend_fcall_info_cache fcc;
3083
} spl_iterator_apply_info;
3084
3085
static int spl_iterator_func_apply(zend_object_iterator *iter, void *puser) /* {{{ */
3086
0
{
3087
0
  zval retval;
3088
0
  spl_iterator_apply_info  *apply_info = (spl_iterator_apply_info*)puser;
3089
0
  int result;
3090
3091
0
  apply_info->count++;
3092
0
  zend_call_known_fcc(&apply_info->fcc, &retval, 0, NULL, apply_info->params_ht);
3093
0
  result = zend_is_true(&retval) ? ZEND_HASH_APPLY_KEEP : ZEND_HASH_APPLY_STOP;
3094
0
  zval_ptr_dtor(&retval);
3095
0
  return result;
3096
0
}
3097
/* }}} */
3098
3099
/* {{{ Calls a function for every element in an iterator */
3100
PHP_FUNCTION(iterator_apply)
3101
6
{
3102
6
  zval *traversable;
3103
6
  zend_fcall_info dummy_fci;
3104
6
  spl_iterator_apply_info apply_info = {
3105
6
    .count = 0,
3106
6
    .params_ht = NULL,
3107
6
    .fcc = { 0 },
3108
6
  };
3109
3110
  /* The HashTable is used to determine positional arguments */
3111
6
  if (zend_parse_parameters(ZEND_NUM_ARGS(), "OF|h!", &traversable, zend_ce_traversable,
3112
6
      &dummy_fci, &apply_info.fcc, &apply_info.params_ht) == FAILURE) {
3113
6
    zend_release_fcall_info_cache(&apply_info.fcc);
3114
6
    RETURN_THROWS();
3115
6
  }
3116
3117
0
  if (spl_iterator_apply(traversable, spl_iterator_func_apply, (void*)&apply_info) == FAILURE) {
3118
0
    zend_release_fcall_info_cache(&apply_info.fcc);
3119
0
    RETURN_THROWS();
3120
0
  }
3121
0
  zend_release_fcall_info_cache(&apply_info.fcc);
3122
0
  RETURN_LONG(apply_info.count);
3123
0
}
3124
/* }}} */
3125
3126
/* {{{ PHP_MINIT_FUNCTION(spl_iterators) */
3127
PHP_MINIT_FUNCTION(spl_iterators)
3128
16
{
3129
16
  spl_ce_RecursiveIterator = register_class_RecursiveIterator(zend_ce_iterator);
3130
3131
16
  spl_ce_OuterIterator = register_class_OuterIterator(zend_ce_iterator);
3132
3133
16
  spl_ce_RecursiveIteratorIterator = register_class_RecursiveIteratorIterator(spl_ce_OuterIterator);
3134
16
  spl_ce_RecursiveIteratorIterator->create_object = spl_RecursiveIteratorIterator_new;
3135
16
  spl_ce_RecursiveIteratorIterator->default_object_handlers = &spl_handlers_rec_it_it;
3136
16
  spl_ce_RecursiveIteratorIterator->get_iterator = spl_recursive_it_get_iterator;
3137
3138
16
  memcpy(&spl_handlers_rec_it_it, &std_object_handlers, sizeof(zend_object_handlers));
3139
16
  spl_handlers_rec_it_it.offset = XtOffsetOf(spl_recursive_it_object, std);
3140
16
  spl_handlers_rec_it_it.get_method = spl_recursive_it_get_method;
3141
16
  spl_handlers_rec_it_it.clone_obj = NULL;
3142
16
  spl_handlers_rec_it_it.free_obj = spl_RecursiveIteratorIterator_free_storage;
3143
16
  spl_handlers_rec_it_it.get_gc = spl_RecursiveIteratorIterator_get_gc;
3144
3145
16
  memcpy(&spl_handlers_dual_it, &std_object_handlers, sizeof(zend_object_handlers));
3146
16
  spl_handlers_dual_it.offset = XtOffsetOf(spl_dual_it_object, std);
3147
16
  spl_handlers_dual_it.get_method = spl_dual_it_get_method;
3148
16
  spl_handlers_dual_it.clone_obj = NULL;
3149
16
  spl_handlers_dual_it.free_obj = spl_dual_it_free_storage;
3150
16
  spl_handlers_dual_it.get_gc = spl_dual_it_get_gc;
3151
3152
16
  spl_ce_IteratorIterator = register_class_IteratorIterator(spl_ce_OuterIterator);
3153
16
  spl_ce_IteratorIterator->create_object = spl_dual_it_new;
3154
16
  spl_ce_IteratorIterator->default_object_handlers = &spl_handlers_dual_it;
3155
3156
16
  spl_ce_FilterIterator = register_class_FilterIterator(spl_ce_IteratorIterator);
3157
16
  spl_ce_FilterIterator->create_object = spl_dual_it_new;
3158
3159
16
  spl_ce_RecursiveFilterIterator = register_class_RecursiveFilterIterator(spl_ce_FilterIterator, spl_ce_RecursiveIterator);
3160
16
  spl_ce_RecursiveFilterIterator->create_object = spl_dual_it_new;
3161
3162
16
  spl_ce_CallbackFilterIterator = register_class_CallbackFilterIterator(spl_ce_FilterIterator);
3163
16
  spl_ce_CallbackFilterIterator->create_object = spl_dual_it_new;
3164
3165
16
  spl_ce_RecursiveCallbackFilterIterator = register_class_RecursiveCallbackFilterIterator(spl_ce_CallbackFilterIterator, spl_ce_RecursiveIterator);
3166
16
  spl_ce_RecursiveCallbackFilterIterator->create_object = spl_dual_it_new;
3167
3168
16
  spl_ce_ParentIterator = register_class_ParentIterator(spl_ce_RecursiveFilterIterator);
3169
16
  spl_ce_ParentIterator->create_object = spl_dual_it_new;
3170
3171
16
  spl_ce_SeekableIterator = register_class_SeekableIterator(zend_ce_iterator);
3172
3173
16
  spl_ce_LimitIterator = register_class_LimitIterator(spl_ce_IteratorIterator);
3174
16
  spl_ce_LimitIterator->create_object = spl_dual_it_new;
3175
3176
16
  spl_ce_CachingIterator = register_class_CachingIterator(spl_ce_IteratorIterator, zend_ce_arrayaccess, zend_ce_countable, zend_ce_stringable);
3177
16
  spl_ce_CachingIterator->create_object = spl_dual_it_new;
3178
3179
16
  spl_ce_RecursiveCachingIterator = register_class_RecursiveCachingIterator(spl_ce_CachingIterator, spl_ce_RecursiveIterator);
3180
16
  spl_ce_RecursiveCachingIterator->create_object = spl_dual_it_new;
3181
3182
16
  spl_ce_NoRewindIterator = register_class_NoRewindIterator(spl_ce_IteratorIterator);
3183
16
  spl_ce_NoRewindIterator->create_object = spl_dual_it_new;
3184
3185
16
  spl_ce_AppendIterator = register_class_AppendIterator(spl_ce_IteratorIterator);
3186
16
  spl_ce_AppendIterator->create_object = spl_dual_it_new;
3187
3188
16
  spl_ce_InfiniteIterator = register_class_InfiniteIterator(spl_ce_IteratorIterator);
3189
16
  spl_ce_InfiniteIterator->create_object = spl_dual_it_new;
3190
3191
16
  spl_ce_RegexIterator = register_class_RegexIterator(spl_ce_FilterIterator);
3192
16
  spl_ce_RegexIterator->create_object = spl_dual_it_new;
3193
3194
16
  spl_ce_RecursiveRegexIterator = register_class_RecursiveRegexIterator(spl_ce_RegexIterator, spl_ce_RecursiveIterator);
3195
16
  spl_ce_RecursiveRegexIterator->create_object = spl_dual_it_new;
3196
3197
16
  spl_ce_EmptyIterator = register_class_EmptyIterator(zend_ce_iterator);
3198
3199
16
  spl_ce_RecursiveTreeIterator = register_class_RecursiveTreeIterator(spl_ce_RecursiveIteratorIterator);
3200
16
  spl_ce_RecursiveTreeIterator->create_object = spl_RecursiveTreeIterator_new;
3201
3202
16
  return SUCCESS;
3203
16
}
3204
/* }}} */