1999-04-08 02:10:10 +08:00
|
|
|
/*
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| Zend Engine |
|
|
|
|
+----------------------------------------------------------------------+
|
2019-01-30 17:23:29 +08:00
|
|
|
| Copyright (c) Zend Technologies Ltd. (http://www.zend.com) |
|
1999-04-08 02:10:10 +08:00
|
|
|
+----------------------------------------------------------------------+
|
2001-12-11 23:16:21 +08:00
|
|
|
| This source file is subject to version 2.00 of the Zend license, |
|
2015-01-03 17:22:58 +08:00
|
|
|
| that is bundled with this package in the file LICENSE, and is |
|
2003-06-11 04:04:29 +08:00
|
|
|
| available through the world-wide-web at the following url: |
|
2001-12-11 23:16:21 +08:00
|
|
|
| http://www.zend.com/license/2_00.txt. |
|
1999-07-16 22:58:16 +08:00
|
|
|
| If you did not receive a copy of the Zend license and are unable to |
|
|
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
|
|
| license@zend.com so we can mail you a copy immediately. |
|
1999-04-08 02:10:10 +08:00
|
|
|
+----------------------------------------------------------------------+
|
2018-11-01 23:20:07 +08:00
|
|
|
| Authors: Andi Gutmans <andi@php.net> |
|
|
|
|
| Zeev Suraski <zeev@php.net> |
|
1999-04-08 02:10:10 +08:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
*/
|
|
|
|
|
2000-07-03 07:54:19 +08:00
|
|
|
#ifndef ZEND_PTR_STACK_H
|
|
|
|
#define ZEND_PTR_STACK_H
|
1999-04-08 02:10:10 +08:00
|
|
|
|
1999-05-23 00:10:51 +08:00
|
|
|
typedef struct _zend_ptr_stack {
|
2000-03-06 13:26:39 +08:00
|
|
|
int top, max;
|
1999-04-08 02:10:10 +08:00
|
|
|
void **elements;
|
|
|
|
void **top_element;
|
2021-01-15 19:30:54 +08:00
|
|
|
bool persistent;
|
1999-04-08 02:10:10 +08:00
|
|
|
} zend_ptr_stack;
|
|
|
|
|
|
|
|
|
1999-07-30 19:55:53 +08:00
|
|
|
#define PTR_STACK_BLOCK_SIZE 64
|
1999-04-08 02:10:10 +08:00
|
|
|
|
2004-02-19 06:44:40 +08:00
|
|
|
BEGIN_EXTERN_C()
|
1999-04-08 02:10:10 +08:00
|
|
|
ZEND_API void zend_ptr_stack_init(zend_ptr_stack *stack);
|
2021-01-15 19:30:54 +08:00
|
|
|
ZEND_API void zend_ptr_stack_init_ex(zend_ptr_stack *stack, bool persistent);
|
1999-12-26 07:52:00 +08:00
|
|
|
ZEND_API void zend_ptr_stack_n_push(zend_ptr_stack *stack, int count, ...);
|
|
|
|
ZEND_API void zend_ptr_stack_n_pop(zend_ptr_stack *stack, int count, ...);
|
1999-04-08 02:10:10 +08:00
|
|
|
ZEND_API void zend_ptr_stack_destroy(zend_ptr_stack *stack);
|
|
|
|
ZEND_API void zend_ptr_stack_apply(zend_ptr_stack *stack, void (*func)(void *));
|
2017-09-15 21:07:59 +08:00
|
|
|
ZEND_API void zend_ptr_stack_reverse_apply(zend_ptr_stack *stack, void (*func)(void *));
|
2021-01-15 19:30:54 +08:00
|
|
|
ZEND_API void zend_ptr_stack_clean(zend_ptr_stack *stack, void (*func)(void *), bool free_elements);
|
2000-06-18 02:04:58 +08:00
|
|
|
ZEND_API int zend_ptr_stack_num_elements(zend_ptr_stack *stack);
|
2004-07-31 05:00:37 +08:00
|
|
|
END_EXTERN_C()
|
2004-07-31 04:16:40 +08:00
|
|
|
|
2004-07-31 05:00:37 +08:00
|
|
|
#define ZEND_PTR_STACK_RESIZE_IF_NEEDED(stack, count) \
|
|
|
|
if (stack->top+count > stack->max) { \
|
|
|
|
/* we need to allocate more memory */ \
|
2010-07-09 15:31:18 +08:00
|
|
|
do { \
|
|
|
|
stack->max += PTR_STACK_BLOCK_SIZE; \
|
|
|
|
} while (stack->top+count > stack->max); \
|
2022-07-04 17:59:16 +08:00
|
|
|
stack->elements = (void **) safe_perealloc(stack->elements, sizeof(void *), (stack->max), 0, stack->persistent); \
|
2004-07-31 05:00:37 +08:00
|
|
|
stack->top_element = stack->elements+stack->top; \
|
|
|
|
}
|
2004-07-31 04:16:40 +08:00
|
|
|
|
|
|
|
/* Not doing this with a macro because of the loop unrolling in the element assignment.
|
|
|
|
Just using a macro for 3 in the body for readability sake. */
|
2010-04-20 19:16:39 +08:00
|
|
|
static zend_always_inline void zend_ptr_stack_3_push(zend_ptr_stack *stack, void *a, void *b, void *c)
|
2004-07-31 04:16:40 +08:00
|
|
|
{
|
|
|
|
#define ZEND_PTR_STACK_NUM_ARGS 3
|
|
|
|
|
2004-07-31 05:00:37 +08:00
|
|
|
ZEND_PTR_STACK_RESIZE_IF_NEEDED(stack, ZEND_PTR_STACK_NUM_ARGS)
|
2004-07-31 04:16:40 +08:00
|
|
|
|
|
|
|
stack->top += ZEND_PTR_STACK_NUM_ARGS;
|
|
|
|
*(stack->top_element++) = a;
|
|
|
|
*(stack->top_element++) = b;
|
|
|
|
*(stack->top_element++) = c;
|
|
|
|
|
|
|
|
#undef ZEND_PTR_STACK_NUM_ARGS
|
|
|
|
}
|
|
|
|
|
2010-04-20 19:16:39 +08:00
|
|
|
static zend_always_inline void zend_ptr_stack_2_push(zend_ptr_stack *stack, void *a, void *b)
|
2004-07-31 04:16:40 +08:00
|
|
|
{
|
2004-07-31 05:00:37 +08:00
|
|
|
#define ZEND_PTR_STACK_NUM_ARGS 2
|
|
|
|
|
|
|
|
ZEND_PTR_STACK_RESIZE_IF_NEEDED(stack, ZEND_PTR_STACK_NUM_ARGS)
|
|
|
|
|
|
|
|
stack->top += ZEND_PTR_STACK_NUM_ARGS;
|
|
|
|
*(stack->top_element++) = a;
|
|
|
|
*(stack->top_element++) = b;
|
|
|
|
|
|
|
|
#undef ZEND_PTR_STACK_NUM_ARGS
|
2004-07-31 04:16:40 +08:00
|
|
|
}
|
|
|
|
|
2010-04-20 19:16:39 +08:00
|
|
|
static zend_always_inline void zend_ptr_stack_3_pop(zend_ptr_stack *stack, void **a, void **b, void **c)
|
2004-07-31 05:00:37 +08:00
|
|
|
{
|
|
|
|
*a = *(--stack->top_element);
|
|
|
|
*b = *(--stack->top_element);
|
|
|
|
*c = *(--stack->top_element);
|
2005-10-20 15:23:26 +08:00
|
|
|
stack->top -= 3;
|
|
|
|
}
|
|
|
|
|
2010-04-20 19:16:39 +08:00
|
|
|
static zend_always_inline void zend_ptr_stack_2_pop(zend_ptr_stack *stack, void **a, void **b)
|
2005-10-20 15:23:26 +08:00
|
|
|
{
|
|
|
|
*a = *(--stack->top_element);
|
|
|
|
*b = *(--stack->top_element);
|
|
|
|
stack->top -= 2;
|
2004-07-31 05:00:37 +08:00
|
|
|
}
|
1999-04-08 02:10:10 +08:00
|
|
|
|
2010-04-20 19:16:39 +08:00
|
|
|
static zend_always_inline void zend_ptr_stack_push(zend_ptr_stack *stack, void *ptr)
|
2001-07-12 01:18:22 +08:00
|
|
|
{
|
2004-07-31 05:00:37 +08:00
|
|
|
ZEND_PTR_STACK_RESIZE_IF_NEEDED(stack, 1)
|
|
|
|
|
2001-07-12 01:18:22 +08:00
|
|
|
stack->top++;
|
|
|
|
*(stack->top_element++) = ptr;
|
|
|
|
}
|
|
|
|
|
2010-04-20 19:16:39 +08:00
|
|
|
static zend_always_inline void *zend_ptr_stack_pop(zend_ptr_stack *stack)
|
2001-07-12 01:18:22 +08:00
|
|
|
{
|
|
|
|
stack->top--;
|
|
|
|
return *(--stack->top_element);
|
|
|
|
}
|
|
|
|
|
2014-09-18 08:59:01 +08:00
|
|
|
static zend_always_inline void *zend_ptr_stack_top(zend_ptr_stack *stack)
|
2012-03-31 02:41:44 +08:00
|
|
|
{
|
|
|
|
return stack->elements[stack->top - 1];
|
|
|
|
}
|
|
|
|
|
2000-07-03 07:54:19 +08:00
|
|
|
#endif /* ZEND_PTR_STACK_H */
|