32# include <emmintrin.h>
35# include <immintrin.h>
37#if defined(__aarch64__) || defined(_M_ARM64)
42# define ZEND_ENDIAN_LOHI(lo, hi) hi; lo;
43# define ZEND_ENDIAN_LOHI_3(lo, mi, hi) hi; mi; lo;
44# define ZEND_ENDIAN_LOHI_4(a, b, c, d) d; c; b; a;
45# define ZEND_ENDIAN_LOHI_C(lo, hi) hi, lo
46# define ZEND_ENDIAN_LOHI_C_3(lo, mi, hi) hi, mi, lo,
47# define ZEND_ENDIAN_LOHI_C_4(a, b, c, d) d, c, b, a
49# define ZEND_ENDIAN_LOHI(lo, hi) lo; hi;
50# define ZEND_ENDIAN_LOHI_3(lo, mi, hi) lo; mi; hi;
51# define ZEND_ENDIAN_LOHI_4(a, b, c, d) a; b; c; d;
52# define ZEND_ENDIAN_LOHI_C(lo, hi) lo, hi
53# define ZEND_ENDIAN_LOHI_C_3(lo, mi, hi) lo, mi, hi,
54# define ZEND_ENDIAN_LOHI_C_4(a, b, c, d) a, b, c, d
66#ifdef ZEND_ENABLE_ZVAL_LONG64
68# define ZEND_SIZE_MAX _UI64_MAX
70# define ZEND_SIZE_MAX SIZE_MAX
73# if defined(ZEND_WIN32)
74# define ZEND_SIZE_MAX _UI32_MAX
76# define ZEND_SIZE_MAX SIZE_MAX
81#define ZEND_TLS static TSRM_TLS
82#define ZEND_EXT_TLS TSRM_TLS
84#define ZEND_TLS static
145#define _ZEND_TYPE_EXTRA_FLAGS_SHIFT 25
146#define _ZEND_TYPE_MASK ((1u << 25) - 1)
148#define _ZEND_TYPE_NAME_BIT (1u << 24)
150#define _ZEND_TYPE_LITERAL_NAME_BIT (1u << 23)
151#define _ZEND_TYPE_LIST_BIT (1u << 22)
152#define _ZEND_TYPE_KIND_MASK (_ZEND_TYPE_LIST_BIT|_ZEND_TYPE_NAME_BIT|_ZEND_TYPE_LITERAL_NAME_BIT)
154#define _ZEND_TYPE_ITERABLE_BIT (1u << 21)
156#define _ZEND_TYPE_ARENA_BIT (1u << 20)
158#define _ZEND_TYPE_INTERSECTION_BIT (1u << 19)
160#define _ZEND_TYPE_UNION_BIT (1u << 18)
162#define _ZEND_TYPE_MAY_BE_MASK ((1u << 18) - 1)
164#define _ZEND_TYPE_NULLABLE_BIT 0x2u
166#define ZEND_TYPE_IS_SET(t) \
167 (((t).type_mask & _ZEND_TYPE_MASK) != 0)
171#define ZEND_TYPE_IS_COMPLEX(t) \
172 ((((t).type_mask) & _ZEND_TYPE_KIND_MASK) != 0)
174#define ZEND_TYPE_HAS_NAME(t) \
175 ((((t).type_mask) & _ZEND_TYPE_NAME_BIT) != 0)
177#define ZEND_TYPE_HAS_LITERAL_NAME(t) \
178 ((((t).type_mask) & _ZEND_TYPE_LITERAL_NAME_BIT) != 0)
180#define ZEND_TYPE_HAS_LIST(t) \
181 ((((t).type_mask) & _ZEND_TYPE_LIST_BIT) != 0)
183#define ZEND_TYPE_IS_ITERABLE_FALLBACK(t) \
184 ((((t).type_mask) & _ZEND_TYPE_ITERABLE_BIT) != 0)
186#define ZEND_TYPE_IS_INTERSECTION(t) \
187 ((((t).type_mask) & _ZEND_TYPE_INTERSECTION_BIT) != 0)
189#define ZEND_TYPE_IS_UNION(t) \
190 ((((t).type_mask) & _ZEND_TYPE_UNION_BIT) != 0)
192#define ZEND_TYPE_USES_ARENA(t) \
193 ((((t).type_mask) & _ZEND_TYPE_ARENA_BIT) != 0)
195#define ZEND_TYPE_IS_ONLY_MASK(t) \
196 (ZEND_TYPE_IS_SET(t) && (t).ptr == NULL)
198#define ZEND_TYPE_NAME(t) \
199 ((zend_string *) (t).ptr)
201#define ZEND_TYPE_LITERAL_NAME(t) \
202 ((const char *) (t).ptr)
204#define ZEND_TYPE_LIST(t) \
205 ((zend_type_list *) (t).ptr)
207#define ZEND_TYPE_LIST_SIZE(num_types) \
208 (sizeof(zend_type_list) + ((num_types) - 1) * sizeof(zend_type))
211#define ZEND_TYPE_LIST_FOREACH(list, type_ptr) do { \
212 zend_type *_list = (list)->types; \
213 zend_type *_end = _list + (list)->num_types; \
214 for (; _list < _end; _list++) { \
217#define ZEND_TYPE_LIST_FOREACH_END() \
223#define ZEND_TYPE_FOREACH(type, type_ptr) do { \
224 zend_type *_cur, *_end; \
225 if (ZEND_TYPE_HAS_LIST(type)) { \
226 zend_type_list *_list = ZEND_TYPE_LIST(type); \
227 _cur = _list->types; \
228 _end = _cur + _list->num_types; \
236#define ZEND_TYPE_FOREACH_END() \
237 } while (++_cur < _end); \
240#define ZEND_TYPE_SET_PTR(t, _ptr) \
243#define ZEND_TYPE_SET_PTR_AND_KIND(t, _ptr, kind_bit) do { \
245 (t).type_mask &= ~_ZEND_TYPE_KIND_MASK; \
246 (t).type_mask |= (kind_bit); \
249#define ZEND_TYPE_SET_LIST(t, list) \
250 ZEND_TYPE_SET_PTR_AND_KIND(t, list, _ZEND_TYPE_LIST_BIT)
254#define ZEND_TYPE_FULL_MASK(t) \
257#define ZEND_TYPE_PURE_MASK(t) \
258 ((t).type_mask & _ZEND_TYPE_MAY_BE_MASK)
260#define ZEND_TYPE_FULL_MASK_WITHOUT_NULL(t) \
261 ((t).type_mask & ~_ZEND_TYPE_NULLABLE_BIT)
263#define ZEND_TYPE_PURE_MASK_WITHOUT_NULL(t) \
264 ((t).type_mask & _ZEND_TYPE_MAY_BE_MASK & ~_ZEND_TYPE_NULLABLE_BIT)
266#define ZEND_TYPE_CONTAINS_CODE(t, code) \
267 (((t).type_mask & (1u << (code))) != 0)
269#define ZEND_TYPE_ALLOW_NULL(t) \
270 (((t).type_mask & _ZEND_TYPE_NULLABLE_BIT) != 0)
272#if defined(__cplusplus) && defined(_MSC_VER)
273# define _ZEND_TYPE_PREFIX zend_type
277# define _ZEND_TYPE_PREFIX
280#define ZEND_TYPE_INIT_NONE(extra_flags) \
281 _ZEND_TYPE_PREFIX { NULL, (extra_flags) }
283#define ZEND_TYPE_INIT_MASK(_type_mask) \
284 _ZEND_TYPE_PREFIX { NULL, (_type_mask) }
286#define ZEND_TYPE_INIT_CODE(code, allow_null, extra_flags) \
287 ZEND_TYPE_INIT_MASK(((code) == _IS_BOOL ? MAY_BE_BOOL : ( (code) == IS_ITERABLE ? _ZEND_TYPE_ITERABLE_BIT : ((code) == IS_MIXED ? MAY_BE_ANY : (1 << (code))))) \
288 | ((allow_null) ? _ZEND_TYPE_NULLABLE_BIT : 0) | (extra_flags))
290#define ZEND_TYPE_INIT_PTR(ptr, type_kind, allow_null, extra_flags) \
291 _ZEND_TYPE_PREFIX { (void *) (ptr), \
292 (type_kind) | ((allow_null) ? _ZEND_TYPE_NULLABLE_BIT : 0) | (extra_flags) }
294#define ZEND_TYPE_INIT_PTR_MASK(ptr, type_mask) \
295 _ZEND_TYPE_PREFIX { (void *) (ptr), (type_mask) }
297#define ZEND_TYPE_INIT_UNION(ptr, extra_flags) \
298 _ZEND_TYPE_PREFIX { (void *) (ptr), (_ZEND_TYPE_LIST_BIT|_ZEND_TYPE_UNION_BIT) | (extra_flags) }
300#define ZEND_TYPE_INIT_INTERSECTION(ptr, extra_flags) \
301 _ZEND_TYPE_PREFIX { (void *) (ptr), (_ZEND_TYPE_LIST_BIT|_ZEND_TYPE_INTERSECTION_BIT) | (extra_flags) }
303#define ZEND_TYPE_INIT_CLASS(class_name, allow_null, extra_flags) \
304 ZEND_TYPE_INIT_PTR(class_name, _ZEND_TYPE_NAME_BIT, allow_null, extra_flags)
306#define ZEND_TYPE_INIT_CLASS_MASK(class_name, type_mask) \
307 ZEND_TYPE_INIT_PTR_MASK(class_name, _ZEND_TYPE_NAME_BIT | (type_mask))
309#define ZEND_TYPE_INIT_CLASS_CONST(class_name, allow_null, extra_flags) \
310 ZEND_TYPE_INIT_PTR(class_name, _ZEND_TYPE_LITERAL_NAME_BIT, allow_null, extra_flags)
312#define ZEND_TYPE_INIT_CLASS_CONST_MASK(class_name, type_mask) \
313 ZEND_TYPE_INIT_PTR_MASK(class_name, (_ZEND_TYPE_LITERAL_NAME_BIT | (type_mask)))
395 uint8_t nIteratorsCount,
429#define HT_INVALID_IDX ((uint32_t) -1)
431#define HT_MIN_MASK ((uint32_t) -2)
441#if SIZEOF_SIZE_T == 4
442# define HT_MAX_SIZE 0x02000000
443# define HT_HASH_TO_BUCKET_EX(data, idx) \
444 ((Bucket*)((char*)(data) + (idx)))
445# define HT_IDX_TO_HASH(idx) \
446 ((idx) * sizeof(Bucket))
447# define HT_HASH_TO_IDX(idx) \
448 ((idx) / sizeof(Bucket))
449#elif SIZEOF_SIZE_T == 8
450# define HT_MAX_SIZE 0x40000000
451# define HT_HASH_TO_BUCKET_EX(data, idx) \
453# define HT_IDX_TO_HASH(idx) \
455# define HT_HASH_TO_IDX(idx) \
458# error "Unknown SIZEOF_SIZE_T"
461#define HT_HASH_EX(data, idx) \
462 ((uint32_t*)(data))[(int32_t)(idx)]
463#define HT_HASH(ht, idx) \
464 HT_HASH_EX((ht)->arHash, idx)
466#define HT_SIZE_TO_MASK(nTableSize) \
467 ((uint32_t)(-((nTableSize) + (nTableSize))))
468#define HT_HASH_SIZE(nTableMask) \
469 (((size_t)-(uint32_t)(nTableMask)) * sizeof(uint32_t))
470#define HT_DATA_SIZE(nTableSize) \
471 ((size_t)(nTableSize) * sizeof(Bucket))
472#define HT_SIZE_EX(nTableSize, nTableMask) \
473 (HT_DATA_SIZE((nTableSize)) + HT_HASH_SIZE((nTableMask)))
475 HT_SIZE_EX((ht)->nTableSize, (ht)->nTableMask)
476#define HT_USED_SIZE(ht) \
477 (HT_HASH_SIZE((ht)->nTableMask) + ((size_t)(ht)->nNumUsed * sizeof(Bucket)))
478#define HT_PACKED_DATA_SIZE(nTableSize) \
479 ((size_t)(nTableSize) * sizeof(zval))
480#define HT_PACKED_SIZE_EX(nTableSize, nTableMask) \
481 (HT_PACKED_DATA_SIZE((nTableSize)) + HT_HASH_SIZE((nTableMask)))
482#define HT_PACKED_SIZE(ht) \
483 HT_PACKED_SIZE_EX((ht)->nTableSize, (ht)->nTableMask)
484#define HT_PACKED_USED_SIZE(ht) \
485 (HT_HASH_SIZE((ht)->nTableMask) + ((size_t)(ht)->nNumUsed * sizeof(zval)))
487# define HT_HASH_RESET(ht) do { \
488 char *p = (char*)&HT_HASH(ht, (ht)->nTableMask); \
489 size_t size = HT_HASH_SIZE((ht)->nTableMask); \
490 __m256i ymm0 = _mm256_setzero_si256(); \
491 ymm0 = _mm256_cmpeq_epi64(ymm0, ymm0); \
492 ZEND_ASSERT(size >= 64 && ((size & 0x3f) == 0)); \
494 _mm256_storeu_si256((__m256i*)p, ymm0); \
495 _mm256_storeu_si256((__m256i*)(p+32), ymm0); \
498 } while (size != 0); \
500#elif defined(__SSE2__)
501# define HT_HASH_RESET(ht) do { \
502 char *p = (char*)&HT_HASH(ht, (ht)->nTableMask); \
503 size_t size = HT_HASH_SIZE((ht)->nTableMask); \
504 __m128i xmm0 = _mm_setzero_si128(); \
505 xmm0 = _mm_cmpeq_epi8(xmm0, xmm0); \
506 ZEND_ASSERT(size >= 64 && ((size & 0x3f) == 0)); \
508 _mm_storeu_si128((__m128i*)p, xmm0); \
509 _mm_storeu_si128((__m128i*)(p+16), xmm0); \
510 _mm_storeu_si128((__m128i*)(p+32), xmm0); \
511 _mm_storeu_si128((__m128i*)(p+48), xmm0); \
514 } while (size != 0); \
516#elif defined(__aarch64__) || defined(_M_ARM64)
517# define HT_HASH_RESET(ht) do { \
518 char *p = (char*)&HT_HASH(ht, (ht)->nTableMask); \
519 size_t size = HT_HASH_SIZE((ht)->nTableMask); \
520 int32x4_t t = vdupq_n_s32(-1); \
521 ZEND_ASSERT(size >= 64 && ((size & 0x3f) == 0)); \
523 vst1q_s32((int32_t*)p, t); \
524 vst1q_s32((int32_t*)(p+16), t); \
525 vst1q_s32((int32_t*)(p+32), t); \
526 vst1q_s32((int32_t*)(p+48), t); \
529 } while (size != 0); \
532# define HT_HASH_RESET(ht) \
533 memset(&HT_HASH(ht, (ht)->nTableMask), HT_INVALID_IDX, HT_HASH_SIZE((ht)->nTableMask))
535#define HT_HASH_RESET_PACKED(ht) do { \
536 HT_HASH(ht, -2) = HT_INVALID_IDX; \
537 HT_HASH(ht, -1) = HT_INVALID_IDX; \
539#define HT_HASH_TO_BUCKET(ht, idx) \
540 HT_HASH_TO_BUCKET_EX((ht)->arData, idx)
542#define HT_SET_DATA_ADDR(ht, ptr) do { \
543 (ht)->arData = (Bucket*)(((char*)(ptr)) + HT_HASH_SIZE((ht)->nTableMask)); \
545#define HT_GET_DATA_ADDR(ht) \
546 ((char*)((ht)->arData) - HT_HASH_SIZE((ht)->nTableMask))
584#define ZEND_PROPERTY_INFO_SOURCE_FROM_LIST(list) (0x1 | (uintptr_t) (list))
585#define ZEND_PROPERTY_INFO_SOURCE_TO_LIST(list) ((zend_property_info_list *) ((list) & ~0x1))
586#define ZEND_PROPERTY_INFO_SOURCE_IS_LIST(list) ((list) & 0x1)
610#define IS_REFERENCE 10
611#define IS_CONSTANT_AST 11
615#define IS_CALLABLE 12
616#define IS_ITERABLE 13
623#define IS_INDIRECT 12
625#define IS_ALIAS_PTR 14
633#define ZEND_GUARD_PROPERTY_GET (1<<0)
634#define ZEND_GUARD_PROPERTY_SET (1<<1)
635#define ZEND_GUARD_PROPERTY_UNSET (1<<2)
636#define ZEND_GUARD_PROPERTY_ISSET (1<<3)
637#define ZEND_GUARD_PROPERTY_HOOK (1<<4)
638#define ZEND_GUARD_PROPERTY_MASK 31
639#define ZEND_GUARD_RECURSION_DEBUG (1<<5)
640#define ZEND_GUARD_RECURSION_EXPORT (1<<6)
641#define ZEND_GUARD_RECURSION_JSON (1<<7)
643#define ZEND_GUARD_RECURSION_TYPE(t) ZEND_GUARD_RECURSION_ ## t
645#define ZEND_GUARD_IS_RECURSIVE(pg, t) ((*pg & ZEND_GUARD_RECURSION_TYPE(t)) != 0)
646#define ZEND_GUARD_PROTECT_RECURSION(pg, t) *pg |= ZEND_GUARD_RECURSION_TYPE(t)
647#define ZEND_GUARD_UNPROTECT_RECURSION(pg, t) *pg &= ~ZEND_GUARD_RECURSION_TYPE(t)
650 return pz->
u1.
v.type;
653#define ZEND_SAME_FAKE_TYPE(faketype, realtype) ( \
654 (faketype) == (realtype) \
655 || ((faketype) == _IS_BOOL && ((realtype) == IS_TRUE || (realtype) == IS_FALSE)) \
659#define Z_TYPE(zval) zval_get_type(&(zval))
660#define Z_TYPE_P(zval_p) Z_TYPE(*(zval_p))
662#define Z_TYPE_FLAGS(zval) (zval).u1.v.type_flags
663#define Z_TYPE_FLAGS_P(zval_p) Z_TYPE_FLAGS(*(zval_p))
665#define Z_TYPE_EXTRA(zval) (zval).u1.v.u.extra
666#define Z_TYPE_EXTRA_P(zval_p) Z_TYPE_EXTRA(*(zval_p))
668#define Z_TYPE_INFO(zval) (zval).u1.type_info
669#define Z_TYPE_INFO_P(zval_p) Z_TYPE_INFO(*(zval_p))
671#define Z_NEXT(zval) (zval).u2.next
672#define Z_NEXT_P(zval_p) Z_NEXT(*(zval_p))
674#define Z_CACHE_SLOT(zval) (zval).u2.cache_slot
675#define Z_CACHE_SLOT_P(zval_p) Z_CACHE_SLOT(*(zval_p))
677#define Z_LINENO(zval) (zval).u2.lineno
678#define Z_LINENO_P(zval_p) Z_LINENO(*(zval_p))
680#define Z_OPLINE_NUM(zval) (zval).u2.opline_num
681#define Z_OPLINE_NUM_P(zval_p) Z_OPLINE_NUM(*(zval_p))
683#define Z_FE_POS(zval) (zval).u2.fe_pos
684#define Z_FE_POS_P(zval_p) Z_FE_POS(*(zval_p))
686#define Z_FE_ITER(zval) (zval).u2.fe_iter_idx
687#define Z_FE_ITER_P(zval_p) Z_FE_ITER(*(zval_p))
689#define Z_GUARD(zval) (zval).u2.guard
690#define Z_GUARD_P(zval_p) Z_GUARD(*(zval_p))
692#define Z_CONSTANT_FLAGS(zval) (zval).u2.constant_flags
693#define Z_CONSTANT_FLAGS_P(zval_p) Z_CONSTANT_FLAGS(*(zval_p))
695#define Z_EXTRA(zval) (zval).u2.extra
696#define Z_EXTRA_P(zval_p) Z_EXTRA(*(zval_p))
698#define Z_COUNTED(zval) (zval).value.counted
699#define Z_COUNTED_P(zval_p) Z_COUNTED(*(zval_p))
701#define Z_TYPE_MASK 0xff
702#define Z_TYPE_FLAGS_MASK 0xff00
704#define Z_TYPE_FLAGS_SHIFT 8
705#define Z_TYPE_INFO_EXTRA_SHIFT 16
707#define GC_REFCOUNT(p) zend_gc_refcount(&(p)->gc)
708#define GC_SET_REFCOUNT(p, rc) zend_gc_set_refcount(&(p)->gc, rc)
709#define GC_ADDREF(p) zend_gc_addref(&(p)->gc)
710#define GC_DELREF(p) zend_gc_delref(&(p)->gc)
711#define GC_ADDREF_EX(p, rc) zend_gc_addref_ex(&(p)->gc, rc)
712#define GC_DELREF_EX(p, rc) zend_gc_delref_ex(&(p)->gc, rc)
713#define GC_TRY_ADDREF(p) zend_gc_try_addref(&(p)->gc)
714#define GC_TRY_DELREF(p) zend_gc_try_delref(&(p)->gc)
718 zend_refcounted_h *_p = &(p)->gc; \
719 if (zend_gc_delref(_p) == 0) { \
720 rc_dtor_func((zend_refcounted *)_p); \
722 gc_check_possible_root((zend_refcounted *)_p); \
726#define GC_DTOR_NO_REF(p) \
728 zend_refcounted_h *_p = &(p)->gc; \
729 if (zend_gc_delref(_p) == 0) { \
730 rc_dtor_func((zend_refcounted *)_p); \
732 gc_check_possible_root_no_ref((zend_refcounted *)_p); \
736#define GC_TYPE_MASK 0x0000000f
737#define GC_FLAGS_MASK 0x000003f0
738#define GC_INFO_MASK 0xfffffc00
739#define GC_FLAGS_SHIFT 0
740#define GC_INFO_SHIFT 10
754#define GC_TYPE_INFO(p) (p)->gc.u.type_info
755#define GC_TYPE(p) zval_gc_type(GC_TYPE_INFO(p))
756#define GC_FLAGS(p) zval_gc_flags(GC_TYPE_INFO(p))
757#define GC_INFO(p) zval_gc_info(GC_TYPE_INFO(p))
759#define GC_ADD_FLAGS(p, flags) do { \
760 GC_TYPE_INFO(p) |= (flags) << GC_FLAGS_SHIFT; \
762#define GC_DEL_FLAGS(p, flags) do { \
763 GC_TYPE_INFO(p) &= ~((flags) << GC_FLAGS_SHIFT); \
766#define Z_GC_TYPE(zval) GC_TYPE(Z_COUNTED(zval))
767#define Z_GC_TYPE_P(zval_p) Z_GC_TYPE(*(zval_p))
769#define Z_GC_FLAGS(zval) GC_FLAGS(Z_COUNTED(zval))
770#define Z_GC_FLAGS_P(zval_p) Z_GC_FLAGS(*(zval_p))
772#define Z_GC_INFO(zval) GC_INFO(Z_COUNTED(zval))
773#define Z_GC_INFO_P(zval_p) Z_GC_INFO(*(zval_p))
774#define Z_GC_TYPE_INFO(zval) GC_TYPE_INFO(Z_COUNTED(zval))
775#define Z_GC_TYPE_INFO_P(zval_p) Z_GC_TYPE_INFO(*(zval_p))
778#define GC_NOT_COLLECTABLE (1<<4)
779#define GC_PROTECTED (1<<5)
780#define GC_IMMUTABLE (1<<6)
781#define GC_PERSISTENT (1<<7)
782#define GC_PERSISTENT_LOCAL (1<<8)
784#define GC_NULL (IS_NULL | (GC_NOT_COLLECTABLE << GC_FLAGS_SHIFT))
785#define GC_STRING (IS_STRING | (GC_NOT_COLLECTABLE << GC_FLAGS_SHIFT))
786#define GC_ARRAY IS_ARRAY
787#define GC_OBJECT IS_OBJECT
788#define GC_RESOURCE (IS_RESOURCE | (GC_NOT_COLLECTABLE << GC_FLAGS_SHIFT))
789#define GC_REFERENCE (IS_REFERENCE | (GC_NOT_COLLECTABLE << GC_FLAGS_SHIFT))
790#define GC_CONSTANT_AST (IS_CONSTANT_AST | (GC_NOT_COLLECTABLE << GC_FLAGS_SHIFT))
793#define IS_TYPE_REFCOUNTED (1<<0)
794#define IS_TYPE_COLLECTABLE (1<<1)
799# define Z_TYPE_INFO_REFCOUNTED(t) (((t) & Z_TYPE_FLAGS_MASK) != 0)
801# define Z_TYPE_INFO_REFCOUNTED(t) (((t) & (IS_TYPE_REFCOUNTED << Z_TYPE_FLAGS_SHIFT)) != 0)
805#define IS_INTERNED_STRING_EX IS_STRING
807#define IS_STRING_EX (IS_STRING | (IS_TYPE_REFCOUNTED << Z_TYPE_FLAGS_SHIFT))
808#define IS_ARRAY_EX (IS_ARRAY | (IS_TYPE_REFCOUNTED << Z_TYPE_FLAGS_SHIFT) | (IS_TYPE_COLLECTABLE << Z_TYPE_FLAGS_SHIFT))
809#define IS_OBJECT_EX (IS_OBJECT | (IS_TYPE_REFCOUNTED << Z_TYPE_FLAGS_SHIFT) | (IS_TYPE_COLLECTABLE << Z_TYPE_FLAGS_SHIFT))
810#define IS_RESOURCE_EX (IS_RESOURCE | (IS_TYPE_REFCOUNTED << Z_TYPE_FLAGS_SHIFT))
811#define IS_REFERENCE_EX (IS_REFERENCE | (IS_TYPE_REFCOUNTED << Z_TYPE_FLAGS_SHIFT))
813#define IS_CONSTANT_AST_EX (IS_CONSTANT_AST | (IS_TYPE_REFCOUNTED << Z_TYPE_FLAGS_SHIFT))
816#define IS_STR_CLASS_NAME_MAP_PTR GC_PROTECTED
817#define IS_STR_INTERNED GC_IMMUTABLE
818#define IS_STR_PERSISTENT GC_PERSISTENT
819#define IS_STR_PERMANENT (1<<8)
820#define IS_STR_VALID_UTF8 (1<<9)
823#define IS_ARRAY_IMMUTABLE GC_IMMUTABLE
824#define IS_ARRAY_PERSISTENT GC_PERSISTENT
827#define IS_OBJ_WEAKLY_REFERENCED GC_PERSISTENT
828#define IS_OBJ_DESTRUCTOR_CALLED (1<<8)
829#define IS_OBJ_FREE_CALLED (1<<9)
831#define OBJ_FLAGS(obj) GC_FLAGS(obj)
835#define IS_OBJ_LAZY_UNINITIALIZED (1U<<31)
836#define IS_OBJ_LAZY_PROXY (1U<<30)
838#define OBJ_EXTRA_FLAGS(obj) ((obj)->extra_flags)
841#define ZSTR_HAS_CE_CACHE(s) (GC_FLAGS(s) & IS_STR_CLASS_NAME_MAP_PTR)
842#define ZSTR_GET_CE_CACHE(s) ZSTR_GET_CE_CACHE_EX(s, 1)
843#define ZSTR_SET_CE_CACHE(s, ce) ZSTR_SET_CE_CACHE_EX(s, ce, 1)
845#define ZSTR_VALID_CE_CACHE(s) EXPECTED((GC_REFCOUNT(s)-1)/sizeof(void *) < CG(map_ptr_last))
847#define ZSTR_GET_CE_CACHE_EX(s, validate) \
848 ((!(validate) || ZSTR_VALID_CE_CACHE(s)) ? GET_CE_CACHE(GC_REFCOUNT(s)) : NULL)
850#define ZSTR_SET_CE_CACHE_EX(s, ce, validate) do { \
851 if (!(validate) || ZSTR_VALID_CE_CACHE(s)) { \
852 ZEND_ASSERT((validate) || ZSTR_VALID_CE_CACHE(s)); \
853 SET_CE_CACHE(GC_REFCOUNT(s), ce); \
857#define GET_CE_CACHE(ce_cache) \
858 (*(zend_class_entry **)ZEND_MAP_PTR_OFFSET2PTR(ce_cache))
860#define SET_CE_CACHE(ce_cache, ce) do { \
861 *((zend_class_entry **)ZEND_MAP_PTR_OFFSET2PTR(ce_cache)) = ce; \
865#define GC_IS_RECURSIVE(p) \
866 (GC_FLAGS(p) & GC_PROTECTED)
868#define GC_PROTECT_RECURSION(p) do { \
869 GC_ADD_FLAGS(p, GC_PROTECTED); \
872#define GC_UNPROTECT_RECURSION(p) do { \
873 GC_DEL_FLAGS(p, GC_PROTECTED); \
876#define GC_TRY_PROTECT_RECURSION(p) do { \
877 if (!(GC_FLAGS(p) & GC_IMMUTABLE)) GC_PROTECT_RECURSION(p); \
880#define GC_TRY_UNPROTECT_RECURSION(p) do { \
881 if (!(GC_FLAGS(p) & GC_IMMUTABLE)) GC_UNPROTECT_RECURSION(p); \
884#define Z_IS_RECURSIVE(zval) GC_IS_RECURSIVE(Z_COUNTED(zval))
885#define Z_PROTECT_RECURSION(zval) GC_PROTECT_RECURSION(Z_COUNTED(zval))
886#define Z_UNPROTECT_RECURSION(zval) GC_UNPROTECT_RECURSION(Z_COUNTED(zval))
887#define Z_IS_RECURSIVE_P(zv) Z_IS_RECURSIVE(*(zv))
888#define Z_PROTECT_RECURSION_P(zv) Z_PROTECT_RECURSION(*(zv))
889#define Z_UNPROTECT_RECURSION_P(zv) Z_UNPROTECT_RECURSION(*(zv))
891#define ZEND_GUARD_OR_GC_IS_RECURSIVE(pg, t, zobj) \
892 (pg ? ZEND_GUARD_IS_RECURSIVE(pg, t) : GC_IS_RECURSIVE(zobj))
894#define ZEND_GUARD_OR_GC_PROTECT_RECURSION(pg, t, zobj) do { \
896 ZEND_GUARD_PROTECT_RECURSION(pg, t); \
898 GC_PROTECT_RECURSION(zobj); \
902#define ZEND_GUARD_OR_GC_UNPROTECT_RECURSION(pg, t, zobj) do { \
904 ZEND_GUARD_UNPROTECT_RECURSION(pg, t); \
906 GC_UNPROTECT_RECURSION(zobj); \
911#define Z_CONSTANT(zval) (Z_TYPE(zval) == IS_CONSTANT_AST)
912#define Z_CONSTANT_P(zval_p) Z_CONSTANT(*(zval_p))
917#define Z_REFCOUNTED(zval) (Z_TYPE_FLAGS(zval) != 0)
919#define Z_REFCOUNTED(zval) ((Z_TYPE_FLAGS(zval) & IS_TYPE_REFCOUNTED) != 0)
921#define Z_REFCOUNTED_P(zval_p) Z_REFCOUNTED(*(zval_p))
923#define Z_COLLECTABLE(zval) ((Z_TYPE_FLAGS(zval) & IS_TYPE_COLLECTABLE) != 0)
924#define Z_COLLECTABLE_P(zval_p) Z_COLLECTABLE(*(zval_p))
927#define Z_COPYABLE(zval) (Z_TYPE(zval) == IS_ARRAY)
928#define Z_COPYABLE_P(zval_p) Z_COPYABLE(*(zval_p))
931#define Z_IMMUTABLE(zval) (Z_TYPE_INFO(zval) == IS_ARRAY)
932#define Z_IMMUTABLE_P(zval_p) Z_IMMUTABLE(*(zval_p))
933#define Z_OPT_IMMUTABLE(zval) Z_IMMUTABLE(zval_p)
934#define Z_OPT_IMMUTABLE_P(zval_p) Z_IMMUTABLE(*(zval_p))
937#define Z_OPT_TYPE(zval) (Z_TYPE_INFO(zval) & Z_TYPE_MASK)
938#define Z_OPT_TYPE_P(zval_p) Z_OPT_TYPE(*(zval_p))
940#define Z_OPT_CONSTANT(zval) (Z_OPT_TYPE(zval) == IS_CONSTANT_AST)
941#define Z_OPT_CONSTANT_P(zval_p) Z_OPT_CONSTANT(*(zval_p))
943#define Z_OPT_REFCOUNTED(zval) Z_TYPE_INFO_REFCOUNTED(Z_TYPE_INFO(zval))
944#define Z_OPT_REFCOUNTED_P(zval_p) Z_OPT_REFCOUNTED(*(zval_p))
947#define Z_OPT_COPYABLE(zval) (Z_OPT_TYPE(zval) == IS_ARRAY)
948#define Z_OPT_COPYABLE_P(zval_p) Z_OPT_COPYABLE(*(zval_p))
950#define Z_OPT_ISREF(zval) (Z_OPT_TYPE(zval) == IS_REFERENCE)
951#define Z_OPT_ISREF_P(zval_p) Z_OPT_ISREF(*(zval_p))
953#define Z_ISREF(zval) (Z_TYPE(zval) == IS_REFERENCE)
954#define Z_ISREF_P(zval_p) Z_ISREF(*(zval_p))
956#define Z_ISUNDEF(zval) (Z_TYPE(zval) == IS_UNDEF)
957#define Z_ISUNDEF_P(zval_p) Z_ISUNDEF(*(zval_p))
959#define Z_ISNULL(zval) (Z_TYPE(zval) == IS_NULL)
960#define Z_ISNULL_P(zval_p) Z_ISNULL(*(zval_p))
962#define Z_ISERROR(zval) (Z_TYPE(zval) == _IS_ERROR)
963#define Z_ISERROR_P(zval_p) Z_ISERROR(*(zval_p))
965#define Z_LVAL(zval) (zval).value.lval
966#define Z_LVAL_P(zval_p) Z_LVAL(*(zval_p))
968#define Z_DVAL(zval) (zval).value.dval
969#define Z_DVAL_P(zval_p) Z_DVAL(*(zval_p))
971#define Z_STR(zval) (zval).value.str
972#define Z_STR_P(zval_p) Z_STR(*(zval_p))
974#define Z_STRVAL(zval) ZSTR_VAL(Z_STR(zval))
975#define Z_STRVAL_P(zval_p) Z_STRVAL(*(zval_p))
977#define Z_STRLEN(zval) ZSTR_LEN(Z_STR(zval))
978#define Z_STRLEN_P(zval_p) Z_STRLEN(*(zval_p))
980#define Z_STRHASH(zval) ZSTR_HASH(Z_STR(zval))
981#define Z_STRHASH_P(zval_p) Z_STRHASH(*(zval_p))
983#define Z_ARR(zval) (zval).value.arr
984#define Z_ARR_P(zval_p) Z_ARR(*(zval_p))
986#define Z_ARRVAL(zval) Z_ARR(zval)
987#define Z_ARRVAL_P(zval_p) Z_ARRVAL(*(zval_p))
989#define Z_OBJ(zval) (zval).value.obj
990#define Z_OBJ_P(zval_p) Z_OBJ(*(zval_p))
992#define Z_OBJ_HT(zval) Z_OBJ(zval)->handlers
993#define Z_OBJ_HT_P(zval_p) Z_OBJ_HT(*(zval_p))
995#define Z_OBJ_HANDLER(zval, hf) Z_OBJ_HT((zval))->hf
996#define Z_OBJ_HANDLER_P(zv_p, hf) Z_OBJ_HANDLER(*(zv_p), hf)
998#define Z_OBJ_HANDLE(zval) (Z_OBJ((zval)))->handle
999#define Z_OBJ_HANDLE_P(zval_p) Z_OBJ_HANDLE(*(zval_p))
1001#define Z_OBJCE(zval) (Z_OBJ(zval)->ce)
1002#define Z_OBJCE_P(zval_p) Z_OBJCE(*(zval_p))
1004#define Z_OBJPROP(zval) Z_OBJ_HT((zval))->get_properties(Z_OBJ(zval))
1005#define Z_OBJPROP_P(zval_p) Z_OBJPROP(*(zval_p))
1007#define Z_RES(zval) (zval).value.res
1008#define Z_RES_P(zval_p) Z_RES(*zval_p)
1010#define Z_RES_HANDLE(zval) Z_RES(zval)->handle
1011#define Z_RES_HANDLE_P(zval_p) Z_RES_HANDLE(*zval_p)
1013#define Z_RES_TYPE(zval) Z_RES(zval)->type
1014#define Z_RES_TYPE_P(zval_p) Z_RES_TYPE(*zval_p)
1016#define Z_RES_VAL(zval) Z_RES(zval)->ptr
1017#define Z_RES_VAL_P(zval_p) Z_RES_VAL(*zval_p)
1019#define Z_REF(zval) (zval).value.ref
1020#define Z_REF_P(zval_p) Z_REF(*(zval_p))
1022#define Z_REFVAL(zval) &Z_REF(zval)->val
1023#define Z_REFVAL_P(zval_p) Z_REFVAL(*(zval_p))
1025#define Z_AST(zval) (zval).value.ast
1026#define Z_AST_P(zval_p) Z_AST(*(zval_p))
1028#define GC_AST(p) ((zend_ast*)(((char*)p) + sizeof(zend_ast_ref)))
1030#define Z_ASTVAL(zval) GC_AST(Z_AST(zval))
1031#define Z_ASTVAL_P(zval_p) Z_ASTVAL(*(zval_p))
1033#define Z_INDIRECT(zval) (zval).value.zv
1034#define Z_INDIRECT_P(zval_p) Z_INDIRECT(*(zval_p))
1036#define Z_CE(zval) (zval).value.ce
1037#define Z_CE_P(zval_p) Z_CE(*(zval_p))
1039#define Z_FUNC(zval) (zval).value.func
1040#define Z_FUNC_P(zval_p) Z_FUNC(*(zval_p))
1042#define Z_PTR(zval) (zval).value.ptr
1043#define Z_PTR_P(zval_p) Z_PTR(*(zval_p))
1045#define ZVAL_UNDEF(z) do { \
1046 Z_TYPE_INFO_P(z) = IS_UNDEF; \
1049#define ZVAL_NULL(z) do { \
1050 Z_TYPE_INFO_P(z) = IS_NULL; \
1053#define ZVAL_FALSE(z) do { \
1054 Z_TYPE_INFO_P(z) = IS_FALSE; \
1057#define ZVAL_TRUE(z) do { \
1058 Z_TYPE_INFO_P(z) = IS_TRUE; \
1061#define ZVAL_BOOL(z, b) do { \
1062 Z_TYPE_INFO_P(z) = \
1063 (b) ? IS_TRUE : IS_FALSE; \
1066#define ZVAL_LONG(z, l) do { \
1068 Z_LVAL_P(__z) = l; \
1069 Z_TYPE_INFO_P(__z) = IS_LONG; \
1072#define ZVAL_DOUBLE(z, d) do { \
1074 Z_DVAL_P(__z) = d; \
1075 Z_TYPE_INFO_P(__z) = IS_DOUBLE; \
1078#define ZVAL_STR(z, s) do { \
1080 zend_string *__s = (s); \
1081 Z_STR_P(__z) = __s; \
1083 Z_TYPE_INFO_P(__z) = ZSTR_IS_INTERNED(__s) ? \
1084 IS_INTERNED_STRING_EX : \
1088#define ZVAL_INTERNED_STR(z, s) do { \
1090 zend_string *__s = (s); \
1091 Z_STR_P(__z) = __s; \
1092 Z_TYPE_INFO_P(__z) = IS_INTERNED_STRING_EX; \
1095#define ZVAL_NEW_STR(z, s) do { \
1097 zend_string *__s = (s); \
1098 Z_STR_P(__z) = __s; \
1099 Z_TYPE_INFO_P(__z) = IS_STRING_EX; \
1102#define ZVAL_STR_COPY(z, s) do { \
1104 zend_string *__s = (s); \
1105 Z_STR_P(__z) = __s; \
1107 if (ZSTR_IS_INTERNED(__s)) { \
1108 Z_TYPE_INFO_P(__z) = IS_INTERNED_STRING_EX; \
1111 Z_TYPE_INFO_P(__z) = IS_STRING_EX; \
1115#define ZVAL_ARR(z, a) do { \
1116 zend_array *__arr = (a); \
1118 Z_ARR_P(__z) = __arr; \
1119 Z_TYPE_INFO_P(__z) = IS_ARRAY_EX; \
1122#define ZVAL_NEW_PERSISTENT_ARR(z) do { \
1124 zend_array *_arr = \
1125 (zend_array *) malloc(sizeof(zend_array)); \
1126 Z_ARR_P(__z) = _arr; \
1127 Z_TYPE_INFO_P(__z) = IS_ARRAY_EX; \
1130#define ZVAL_OBJ(z, o) do { \
1132 Z_OBJ_P(__z) = (o); \
1133 Z_TYPE_INFO_P(__z) = IS_OBJECT_EX; \
1136#define ZVAL_OBJ_COPY(z, o) do { \
1138 zend_object *__o = (o); \
1140 Z_OBJ_P(__z) = __o; \
1141 Z_TYPE_INFO_P(__z) = IS_OBJECT_EX; \
1144#define ZVAL_RES(z, r) do { \
1146 Z_RES_P(__z) = (r); \
1147 Z_TYPE_INFO_P(__z) = IS_RESOURCE_EX; \
1150#define ZVAL_NEW_RES(z, h, p, t) do { \
1151 zend_resource *_res = \
1152 (zend_resource *) emalloc(sizeof(zend_resource)); \
1154 GC_SET_REFCOUNT(_res, 1); \
1155 GC_TYPE_INFO(_res) = GC_RESOURCE; \
1156 _res->handle = (h); \
1160 Z_RES_P(__z) = _res; \
1161 Z_TYPE_INFO_P(__z) = IS_RESOURCE_EX; \
1164#define ZVAL_NEW_PERSISTENT_RES(z, h, p, t) do { \
1165 zend_resource *_res = \
1166 (zend_resource *) malloc(sizeof(zend_resource)); \
1168 GC_SET_REFCOUNT(_res, 1); \
1169 GC_TYPE_INFO(_res) = GC_RESOURCE | \
1170 (GC_PERSISTENT << GC_FLAGS_SHIFT); \
1171 _res->handle = (h); \
1175 Z_RES_P(__z) = _res; \
1176 Z_TYPE_INFO_P(__z) = IS_RESOURCE_EX; \
1179#define ZVAL_REF(z, r) do { \
1181 Z_REF_P(__z) = (r); \
1182 Z_TYPE_INFO_P(__z) = IS_REFERENCE_EX; \
1185#define ZVAL_NEW_EMPTY_REF(z) do { \
1186 zend_reference *_ref = \
1187 (zend_reference *) emalloc(sizeof(zend_reference)); \
1188 GC_SET_REFCOUNT(_ref, 1); \
1189 GC_TYPE_INFO(_ref) = GC_REFERENCE; \
1190 _ref->sources.ptr = NULL; \
1191 Z_REF_P(z) = _ref; \
1192 Z_TYPE_INFO_P(z) = IS_REFERENCE_EX; \
1195#define ZVAL_NEW_REF(z, r) do { \
1196 zend_reference *_ref = \
1197 (zend_reference *) emalloc(sizeof(zend_reference)); \
1198 GC_SET_REFCOUNT(_ref, 1); \
1199 GC_TYPE_INFO(_ref) = GC_REFERENCE; \
1200 ZVAL_COPY_VALUE(&_ref->val, r); \
1201 _ref->sources.ptr = NULL; \
1202 Z_REF_P(z) = _ref; \
1203 Z_TYPE_INFO_P(z) = IS_REFERENCE_EX; \
1206#define ZVAL_MAKE_REF_EX(z, refcount) do { \
1208 zend_reference *_ref = \
1209 (zend_reference *) emalloc(sizeof(zend_reference)); \
1210 GC_SET_REFCOUNT(_ref, (refcount)); \
1211 GC_TYPE_INFO(_ref) = GC_REFERENCE; \
1212 ZVAL_COPY_VALUE(&_ref->val, _z); \
1213 _ref->sources.ptr = NULL; \
1214 Z_REF_P(_z) = _ref; \
1215 Z_TYPE_INFO_P(_z) = IS_REFERENCE_EX; \
1218#define ZVAL_NEW_PERSISTENT_REF(z, r) do { \
1219 zend_reference *_ref = \
1220 (zend_reference *) malloc(sizeof(zend_reference)); \
1221 GC_SET_REFCOUNT(_ref, 1); \
1222 GC_TYPE_INFO(_ref) = GC_REFERENCE | \
1223 (GC_PERSISTENT << GC_FLAGS_SHIFT); \
1224 ZVAL_COPY_VALUE(&_ref->val, r); \
1225 _ref->sources.ptr = NULL; \
1226 Z_REF_P(z) = _ref; \
1227 Z_TYPE_INFO_P(z) = IS_REFERENCE_EX; \
1230#define ZVAL_AST(z, ast) do { \
1232 Z_AST_P(__z) = ast; \
1233 Z_TYPE_INFO_P(__z) = IS_CONSTANT_AST_EX; \
1236#define ZVAL_INDIRECT(z, v) do { \
1237 Z_INDIRECT_P(z) = (v); \
1238 Z_TYPE_INFO_P(z) = IS_INDIRECT; \
1241#define ZVAL_PTR(z, p) do { \
1243 Z_TYPE_INFO_P(z) = IS_PTR; \
1246#define ZVAL_FUNC(z, f) do { \
1247 Z_FUNC_P(z) = (f); \
1248 Z_TYPE_INFO_P(z) = IS_PTR; \
1251#define ZVAL_CE(z, c) do { \
1253 Z_TYPE_INFO_P(z) = IS_PTR; \
1256#define ZVAL_ALIAS_PTR(z, p) do { \
1258 Z_TYPE_INFO_P(z) = IS_ALIAS_PTR; \
1261#define ZVAL_ERROR(z) do { \
1262 Z_TYPE_INFO_P(z) = _IS_ERROR; \
1265#define Z_REFCOUNT_P(pz) zval_refcount_p(pz)
1266#define Z_SET_REFCOUNT_P(pz, rc) zval_set_refcount_p(pz, rc)
1267#define Z_ADDREF_P(pz) zval_addref_p(pz)
1268#define Z_DELREF_P(pz) zval_delref_p(pz)
1270#define Z_REFCOUNT(z) Z_REFCOUNT_P(&(z))
1271#define Z_SET_REFCOUNT(z, rc) Z_SET_REFCOUNT_P(&(z), rc)
1272#define Z_ADDREF(z) Z_ADDREF_P(&(z))
1273#define Z_DELREF(z) Z_DELREF_P(&(z))
1275#define Z_TRY_ADDREF_P(pz) do { \
1276 if (Z_REFCOUNTED_P((pz))) { \
1281#define Z_TRY_DELREF_P(pz) do { \
1282 if (Z_REFCOUNTED_P((pz))) { \
1287#define Z_TRY_ADDREF(z) Z_TRY_ADDREF_P(&(z))
1288#define Z_TRY_DELREF(z) Z_TRY_DELREF_P(&(z))
1290#ifndef ZEND_RC_DEBUG
1291# define ZEND_RC_DEBUG 0
1298# define ZEND_RC_MOD_CHECK(p) do { \
1299 if (zend_rc_debug) { \
1300 uint8_t type = zval_gc_type((p)->u.type_info); \
1301 if (type != IS_OBJECT && type != IS_NULL) { \
1302 ZEND_ASSERT(!(zval_gc_flags((p)->u.type_info) & GC_IMMUTABLE)); \
1303 ZEND_ASSERT((zval_gc_flags((p)->u.type_info) & (GC_PERSISTENT|GC_PERSISTENT_LOCAL)) != GC_PERSISTENT); \
1307# define GC_MAKE_PERSISTENT_LOCAL(p) do { \
1308 GC_ADD_FLAGS(p, GC_PERSISTENT_LOCAL); \
1311# define ZEND_RC_MOD_CHECK(p) \
1313# define GC_MAKE_PERSISTENT_LOCAL(p) \
1328 return ++(
p->refcount);
1348 return --(
p->refcount);
1385#if SIZEOF_SIZE_T == 4
1386# define ZVAL_COPY_VALUE_EX(z, v, gc, t) \
1388 uint32_t _w2 = v->value.ww.w2; \
1389 Z_COUNTED_P(z) = gc; \
1390 z->value.ww.w2 = _w2; \
1391 Z_TYPE_INFO_P(z) = t; \
1393#elif SIZEOF_SIZE_T == 8
1394# define ZVAL_COPY_VALUE_EX(z, v, gc, t) \
1396 Z_COUNTED_P(z) = gc; \
1397 Z_TYPE_INFO_P(z) = t; \
1400# error "Unknown SIZEOF_SIZE_T"
1403#define ZVAL_COPY_VALUE(z, v) \
1406 const zval *_z2 = (v); \
1407 zend_refcounted *_gc = Z_COUNTED_P(_z2); \
1408 uint32_t _t = Z_TYPE_INFO_P(_z2); \
1409 ZVAL_COPY_VALUE_EX(_z1, _z2, _gc, _t); \
1412#define ZVAL_COPY(z, v) \
1415 const zval *_z2 = (v); \
1416 zend_refcounted *_gc = Z_COUNTED_P(_z2); \
1417 uint32_t _t = Z_TYPE_INFO_P(_z2); \
1418 ZVAL_COPY_VALUE_EX(_z1, _z2, _gc, _t); \
1419 if (Z_TYPE_INFO_REFCOUNTED(_t)) { \
1424#define ZVAL_DUP(z, v) \
1427 const zval *_z2 = (v); \
1428 zend_refcounted *_gc = Z_COUNTED_P(_z2); \
1429 uint32_t _t = Z_TYPE_INFO_P(_z2); \
1430 if ((_t & Z_TYPE_MASK) == IS_ARRAY) { \
1431 ZVAL_ARR(_z1, zend_array_dup((zend_array*)_gc));\
1433 if (Z_TYPE_INFO_REFCOUNTED(_t)) { \
1436 ZVAL_COPY_VALUE_EX(_z1, _z2, _gc, _t); \
1444#define ZVAL_COPY_OR_DUP(z, v) \
1447 const zval *_z2 = (v); \
1448 zend_refcounted *_gc = Z_COUNTED_P(_z2); \
1449 uint32_t _t = Z_TYPE_INFO_P(_z2); \
1450 ZVAL_COPY_VALUE_EX(_z1, _z2, _gc, _t); \
1451 if (Z_TYPE_INFO_REFCOUNTED(_t)) { \
1453 if (EXPECTED(!(GC_FLAGS(_gc) & GC_PERSISTENT) \
1454 || GC_TYPE(_gc) == IS_OBJECT)) { \
1457 zval_copy_ctor_func(_z1); \
1462#define ZVAL_DEREF(z) do { \
1463 if (UNEXPECTED(Z_ISREF_P(z))) { \
1464 (z) = Z_REFVAL_P(z); \
1468#define ZVAL_DEINDIRECT(z) do { \
1469 if (Z_TYPE_P(z) == IS_INDIRECT) { \
1470 (z) = Z_INDIRECT_P(z); \
1474#define ZVAL_OPT_DEREF(z) do { \
1475 if (UNEXPECTED(Z_OPT_ISREF_P(z))) { \
1476 (z) = Z_REFVAL_P(z); \
1480#define ZVAL_MAKE_REF(zv) do { \
1481 zval *__zv = (zv); \
1482 if (!Z_ISREF_P(__zv)) { \
1483 ZVAL_NEW_REF(__zv, __zv); \
1487#define ZVAL_UNREF(z) do { \
1489 zend_reference *ref; \
1490 ZEND_ASSERT(Z_ISREF_P(_z)); \
1491 ref = Z_REF_P(_z); \
1492 ZVAL_COPY_VALUE(_z, &ref->val); \
1493 efree_size(ref, sizeof(zend_reference)); \
1496#define ZVAL_COPY_DEREF(z, v) do { \
1498 if (Z_OPT_REFCOUNTED_P(_z3)) { \
1499 if (UNEXPECTED(Z_OPT_ISREF_P(_z3))) { \
1500 _z3 = Z_REFVAL_P(_z3); \
1501 if (Z_OPT_REFCOUNTED_P(_z3)) { \
1508 ZVAL_COPY_VALUE(z, _z3); \
1512#define SEPARATE_STRING(zv) do { \
1514 if (Z_REFCOUNT_P(_zv) > 1) { \
1515 zend_string *_str = Z_STR_P(_zv); \
1516 ZEND_ASSERT(Z_REFCOUNTED_P(_zv)); \
1517 ZEND_ASSERT(!ZSTR_IS_INTERNED(_str)); \
1518 ZVAL_NEW_STR(_zv, zend_string_init( \
1519 ZSTR_VAL(_str), ZSTR_LEN(_str), 0)); \
1524#define SEPARATE_ARRAY(zv) do { \
1525 zval *__zv = (zv); \
1526 zend_array *_arr = Z_ARR_P(__zv); \
1527 if (UNEXPECTED(GC_REFCOUNT(_arr) > 1)) { \
1528 ZVAL_ARR(__zv, zend_array_dup(_arr)); \
1529 GC_TRY_DELREF(_arr); \
1533#define SEPARATE_ZVAL_NOREF(zv) do { \
1535 ZEND_ASSERT(Z_TYPE_P(_zv) != IS_REFERENCE); \
1536 if (Z_TYPE_P(_zv) == IS_ARRAY) { \
1537 SEPARATE_ARRAY(_zv); \
1541#define SEPARATE_ZVAL(zv) do { \
1543 if (Z_ISREF_P(_zv)) { \
1544 zend_reference *_r = Z_REF_P(_zv); \
1545 ZVAL_COPY_VALUE(_zv, &_r->val); \
1546 if (GC_DELREF(_r) == 0) { \
1547 efree_size(_r, sizeof(zend_reference)); \
1548 } else if (Z_OPT_TYPE_P(_zv) == IS_ARRAY) { \
1549 ZVAL_ARR(_zv, zend_array_dup(Z_ARR_P(_zv)));\
1551 } else if (Z_OPT_REFCOUNTED_P(_zv)) { \
1556 if (Z_TYPE_P(_zv) == IS_ARRAY) { \
1557 SEPARATE_ARRAY(_zv); \
1565#define IS_PROP_UNINIT (1<<0)
1566#define IS_PROP_REINITABLE (1<<1)
1567#define IS_PROP_LAZY (1<<2)
1568#define Z_PROP_FLAG_P(z) Z_EXTRA_P(z)
1569#define ZVAL_COPY_VALUE_PROP(z, v) \
1570 do { *(z) = *(v); } while (0)
1571#define ZVAL_COPY_PROP(z, v) \
1572 do { ZVAL_COPY(z, v); Z_PROP_FLAG_P(z) = Z_PROP_FLAG_P(v); } while (0)
1573#define ZVAL_COPY_OR_DUP_PROP(z, v) \
1574 do { ZVAL_COPY_OR_DUP(z, v); Z_PROP_FLAG_P(z) = Z_PROP_FLAG_P(v); } while (0)
zend_long nNextFreeElement
uint32_t nInternalPointer
const zend_object_handlers * handlers
zend_property_info_source_list sources
union _zval_struct::@150340241233176113376123270230326260057221363216 u2
struct _zval_struct::@260126255073354372365216353174036005022361071053::@377366240245005141361110052226025141266322210237 v
union _zval_struct::@260126255073354372365216353174036005022361071053 u1
struct _zend_property_info * ptr[1]
struct _zend_value::@155031175007170007274221144332361003151230074200 ww
zend_refcounted * counted
struct _zend_property_info * ptr
ZEND_API void(ZEND_FASTCALL *zend_touch_vm_stack_data)(void *vm_stack_data)
union _zend_function zend_function
struct _zend_string zend_string
#define zend_always_inline
struct _zend_array zend_array
struct _zend_class_entry zend_class_entry
struct _zend_object zend_object
int(* compare_func_t)(const void *, const void *)
struct _zend_ast_ref zend_ast_ref
#define GC_SET_REFCOUNT(p, rc)
#define Z_REFCOUNTED_P(zval_p)
struct _zend_resource zend_resource
struct _zend_array HashTable
union _zend_value zend_value
void(* sort_func_t)(void *, size_t, size_t, compare_func_t, swap_func_t)
#define Z_COUNTED_P(zval_p)
void(* dtor_func_t)(zval *pDest)
#define ZEND_RC_MOD_CHECK(p)
#define ZEND_ENDIAN_LOHI_4(a, b, c, d)
void(* copy_ctor_func_t)(zval *pElement)
struct _zend_refcounted zend_refcounted
void(* swap_func_t)(void *, void *)
struct _zend_refcounted_h zend_refcounted_h
ZEND_RESULT_CODE zend_result
struct _HashTableIterator HashTableIterator
struct _zend_object_handlers zend_object_handlers
struct _zend_ast zend_ast
struct _zend_execute_data zend_execute_data
struct _zend_reference zend_reference
#define ZEND_ENDIAN_LOHI_3(lo, mi, hi)