2006-05-10 07:53:23 +08:00
|
|
|
/*
|
2003-02-01 09:49:15 +08:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| Zend Engine |
|
|
|
|
+----------------------------------------------------------------------+
|
2019-01-30 17:23:29 +08:00
|
|
|
| Copyright (c) Zend Technologies Ltd. (http://www.zend.com) |
|
2003-02-01 09:49:15 +08:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| This source file is subject to version 2.00 of the Zend license, |
|
|
|
|
| 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: |
|
2003-02-01 09:49:15 +08:00
|
|
|
| http://www.zend.com/license/2_00.txt. |
|
|
|
|
| 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. |
|
|
|
|
+----------------------------------------------------------------------+
|
2018-11-01 23:20:07 +08:00
|
|
|
| Authors: Andi Gutmans <andi@php.net> |
|
|
|
|
| Zeev Suraski <zeev@php.net> |
|
2003-02-01 09:49:15 +08:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
*/
|
|
|
|
|
2002-05-31 20:09:19 +08:00
|
|
|
#ifndef ZEND_OBJECTS_API_H
|
|
|
|
#define ZEND_OBJECTS_API_H
|
|
|
|
|
|
|
|
#include "zend.h"
|
2015-02-04 20:24:13 +08:00
|
|
|
#include "zend_compile.h"
|
2002-05-31 20:09:19 +08:00
|
|
|
|
2014-04-09 05:50:15 +08:00
|
|
|
#define OBJ_BUCKET_INVALID (1<<0)
|
|
|
|
|
|
|
|
#define IS_OBJ_VALID(o) (!(((zend_uintptr_t)(o)) & OBJ_BUCKET_INVALID))
|
|
|
|
|
|
|
|
#define SET_OBJ_INVALID(o) ((zend_object*)((((zend_uintptr_t)(o)) | OBJ_BUCKET_INVALID)))
|
|
|
|
|
|
|
|
#define GET_OBJ_BUCKET_NUMBER(o) (((zend_intptr_t)(o)) >> 1)
|
|
|
|
|
|
|
|
#define SET_OBJ_BUCKET_NUMBER(o, n) do { \
|
|
|
|
(o) = (zend_object*)((((zend_uintptr_t)(n)) << 1) | OBJ_BUCKET_INVALID); \
|
2014-02-12 22:08:11 +08:00
|
|
|
} while (0)
|
|
|
|
|
2018-06-11 21:00:09 +08:00
|
|
|
#define ZEND_OBJECTS_STORE_ADD_TO_FREE_LIST(h) do { \
|
|
|
|
SET_OBJ_BUCKET_NUMBER(EG(objects_store).object_buckets[(h)], EG(objects_store).free_list_head); \
|
|
|
|
EG(objects_store).free_list_head = (h); \
|
|
|
|
} while (0)
|
2014-02-14 21:48:45 +08:00
|
|
|
|
2014-12-14 06:06:14 +08:00
|
|
|
#define OBJ_RELEASE(obj) zend_object_release(obj)
|
2014-02-14 21:48:45 +08:00
|
|
|
|
2002-05-31 20:09:19 +08:00
|
|
|
typedef struct _zend_objects_store {
|
2014-02-10 14:04:30 +08:00
|
|
|
zend_object **object_buckets;
|
2014-08-26 01:28:33 +08:00
|
|
|
uint32_t top;
|
|
|
|
uint32_t size;
|
2002-05-31 20:09:19 +08:00
|
|
|
int free_list_head;
|
|
|
|
} zend_objects_store;
|
|
|
|
|
|
|
|
/* Global store handling functions */
|
2004-02-19 06:44:40 +08:00
|
|
|
BEGIN_EXTERN_C()
|
2017-12-14 18:50:39 +08:00
|
|
|
ZEND_API void ZEND_FASTCALL zend_objects_store_init(zend_objects_store *objects, uint32_t init_size);
|
|
|
|
ZEND_API void ZEND_FASTCALL zend_objects_store_call_destructors(zend_objects_store *objects);
|
|
|
|
ZEND_API void ZEND_FASTCALL zend_objects_store_mark_destructed(zend_objects_store *objects);
|
2021-01-15 19:30:54 +08:00
|
|
|
ZEND_API void ZEND_FASTCALL zend_objects_store_free_object_storage(zend_objects_store *objects, bool fast_shutdown);
|
2017-12-14 18:50:39 +08:00
|
|
|
ZEND_API void ZEND_FASTCALL zend_objects_store_destroy(zend_objects_store *objects);
|
2002-05-31 20:09:19 +08:00
|
|
|
|
|
|
|
/* Store API functions */
|
2017-12-14 18:50:39 +08:00
|
|
|
ZEND_API void ZEND_FASTCALL zend_objects_store_put(zend_object *object);
|
|
|
|
ZEND_API void ZEND_FASTCALL zend_objects_store_del(zend_object *object);
|
2016-11-21 18:13:28 +08:00
|
|
|
|
|
|
|
/* Called when the ctor was terminated by an exception */
|
|
|
|
static zend_always_inline void zend_object_store_ctor_failed(zend_object *obj)
|
|
|
|
{
|
2018-02-28 05:52:12 +08:00
|
|
|
GC_ADD_FLAGS(obj, IS_OBJ_DESTRUCTOR_CALLED);
|
2016-11-21 18:13:28 +08:00
|
|
|
}
|
2004-05-17 19:27:16 +08:00
|
|
|
|
2004-02-19 06:44:40 +08:00
|
|
|
END_EXTERN_C()
|
|
|
|
|
2014-12-14 06:06:14 +08:00
|
|
|
static zend_always_inline void zend_object_release(zend_object *obj)
|
2014-10-01 20:35:56 +08:00
|
|
|
{
|
2017-10-27 06:28:58 +08:00
|
|
|
if (GC_DELREF(obj) == 0) {
|
2014-12-14 06:06:14 +08:00
|
|
|
zend_objects_store_del(obj);
|
2016-10-21 22:47:30 +08:00
|
|
|
} else if (UNEXPECTED(GC_MAY_LEAK((zend_refcounted*)obj))) {
|
2015-08-13 18:56:29 +08:00
|
|
|
gc_possible_root((zend_refcounted*)obj);
|
2014-10-01 20:35:56 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-04 20:24:13 +08:00
|
|
|
static zend_always_inline size_t zend_object_properties_size(zend_class_entry *ce)
|
|
|
|
{
|
|
|
|
return sizeof(zval) *
|
|
|
|
(ce->default_properties_count -
|
|
|
|
((ce->ce_flags & ZEND_ACC_USE_GUARDS) ? 0 : 1));
|
|
|
|
}
|
|
|
|
|
2017-11-25 23:07:51 +08:00
|
|
|
/* Allocates object type and zeros it, but not the properties.
|
|
|
|
* Properties MUST be initialized using object_properties_init(). */
|
|
|
|
static zend_always_inline void *zend_object_alloc(size_t obj_size, zend_class_entry *ce) {
|
|
|
|
void *obj = emalloc(obj_size + zend_object_properties_size(ce));
|
2017-11-26 01:02:01 +08:00
|
|
|
/* Subtraction of sizeof(zval) is necessary, because zend_object_properties_size() may be
|
|
|
|
* -sizeof(zval), if the object has no properties. */
|
|
|
|
memset(obj, 0, obj_size - sizeof(zval));
|
2017-11-25 23:07:51 +08:00
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
2019-01-07 19:28:51 +08:00
|
|
|
static inline zend_property_info *zend_get_property_info_for_slot(zend_object *obj, zval *slot)
|
|
|
|
{
|
|
|
|
zend_property_info **table = obj->ce->properties_info_table;
|
|
|
|
intptr_t prop_num = slot - obj->properties_table;
|
|
|
|
ZEND_ASSERT(prop_num >= 0 && prop_num < obj->ce->default_properties_count);
|
|
|
|
return table[prop_num];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper for cases where we're only interested in property info of typed properties. */
|
|
|
|
static inline zend_property_info *zend_get_typed_property_info_for_slot(zend_object *obj, zval *slot)
|
|
|
|
{
|
|
|
|
zend_property_info *prop_info = zend_get_property_info_for_slot(obj, slot);
|
2019-09-23 22:53:54 +08:00
|
|
|
if (prop_info && ZEND_TYPE_IS_SET(prop_info->type)) {
|
2019-01-07 19:28:51 +08:00
|
|
|
return prop_info;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2017-11-25 23:07:51 +08:00
|
|
|
|
2002-05-31 20:09:19 +08:00
|
|
|
#endif /* ZEND_OBJECTS_H */
|