php-src/Zend/zend_variables.c

136 lines
4.6 KiB
C
Raw Normal View History

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 |
| 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
+----------------------------------------------------------------------+
| Authors: Andi Gutmans <andi@php.net> |
| Zeev Suraski <zeev@php.net> |
| Dmitry Stogov <dmitry@php.net> |
1999-04-08 02:10:10 +08:00
+----------------------------------------------------------------------+
*/
#include <stdio.h>
1999-04-08 02:10:10 +08:00
#include "zend.h"
#include "zend_API.h"
#include "zend_ast.h"
1999-04-08 02:10:10 +08:00
#include "zend_globals.h"
#include "zend_constants.h"
#include "zend_list.h"
#if ZEND_DEBUG
static void ZEND_FASTCALL zend_string_destroy(zend_string *str);
#else
# define zend_string_destroy _efree
#endif
static void ZEND_FASTCALL zend_reference_destroy(zend_reference *ref);
static void ZEND_FASTCALL zend_empty_destroy(zend_reference *ref);
2018-01-16 05:57:47 +08:00
typedef void (ZEND_FASTCALL *zend_rc_dtor_func_t)(zend_refcounted *p);
2018-01-16 05:57:47 +08:00
2018-07-06 19:14:44 +08:00
static const zend_rc_dtor_func_t zend_rc_dtor_func[] = {
[IS_UNDEF] = (zend_rc_dtor_func_t)zend_empty_destroy,
[IS_NULL] = (zend_rc_dtor_func_t)zend_empty_destroy,
[IS_FALSE] = (zend_rc_dtor_func_t)zend_empty_destroy,
[IS_TRUE] = (zend_rc_dtor_func_t)zend_empty_destroy,
[IS_LONG] = (zend_rc_dtor_func_t)zend_empty_destroy,
[IS_DOUBLE] = (zend_rc_dtor_func_t)zend_empty_destroy,
[IS_STRING] = (zend_rc_dtor_func_t)zend_string_destroy,
[IS_ARRAY] = (zend_rc_dtor_func_t)zend_array_destroy,
[IS_OBJECT] = (zend_rc_dtor_func_t)zend_objects_store_del,
[IS_RESOURCE] = (zend_rc_dtor_func_t)zend_list_free,
[IS_REFERENCE] = (zend_rc_dtor_func_t)zend_reference_destroy,
[IS_CONSTANT_AST] = (zend_rc_dtor_func_t)zend_ast_ref_destroy
2018-01-16 05:57:47 +08:00
};
ZEND_API void ZEND_FASTCALL rc_dtor_func(zend_refcounted *p)
1999-04-08 02:10:10 +08:00
{
2018-01-16 05:57:47 +08:00
ZEND_ASSERT(GC_TYPE(p) <= IS_CONSTANT_AST);
zend_rc_dtor_func[GC_TYPE(p)](p);
2018-01-16 05:57:47 +08:00
}
2015-01-03 17:22:58 +08:00
#if ZEND_DEBUG
static void ZEND_FASTCALL zend_string_destroy(zend_string *str)
2018-01-16 05:57:47 +08:00
{
CHECK_ZVAL_STRING(str);
2018-01-16 05:57:47 +08:00
ZEND_ASSERT(!ZSTR_IS_INTERNED(str));
ZEND_ASSERT(GC_REFCOUNT(str) == 0);
ZEND_ASSERT(!(GC_FLAGS(str) & IS_STR_PERSISTENT));
efree(str);
2018-01-16 05:57:47 +08:00
}
#endif
2015-01-03 17:22:58 +08:00
static void ZEND_FASTCALL zend_reference_destroy(zend_reference *ref)
2018-01-16 05:57:47 +08:00
{
ZEND_ASSERT(!ZEND_REF_HAS_TYPE_SOURCES(ref));
i_zval_ptr_dtor(&ref->val);
2018-01-16 05:57:47 +08:00
efree_size(ref, sizeof(zend_reference));
}
2015-01-03 17:22:58 +08:00
static void ZEND_FASTCALL zend_empty_destroy(zend_reference *ref)
2018-01-16 05:57:47 +08:00
{
}
2015-01-03 17:22:58 +08:00
ZEND_API void zval_ptr_dtor(zval *zval_ptr) /* {{{ */
{
i_zval_ptr_dtor(zval_ptr);
}
/* }}} */
ZEND_API void zval_internal_ptr_dtor(zval *zval_ptr) /* {{{ */
{
if (Z_REFCOUNTED_P(zval_ptr)) {
zend_refcounted *ref = Z_COUNTED_P(zval_ptr);
if (GC_DELREF(ref) == 0) {
if (Z_TYPE_P(zval_ptr) == IS_STRING) {
zend_string *str = (zend_string*)ref;
CHECK_ZVAL_STRING(str);
ZEND_ASSERT(!ZSTR_IS_INTERNED(str));
ZEND_ASSERT((GC_FLAGS(str) & IS_STR_PERSISTENT));
free(str);
} else {
zend_error_noreturn(E_CORE_ERROR, "Internal zval's can't be arrays, objects, resources or reference");
}
}
}
}
/* }}} */
/* This function should only be used as a copy constructor, i.e. it
* should only be called AFTER a zval has been copied to another
* location using ZVAL_COPY_VALUE. Do not call it before copying,
* otherwise a reference may be leaked. */
ZEND_API void zval_add_ref(zval *p)
1999-04-08 02:10:10 +08:00
{
if (Z_REFCOUNTED_P(p)) {
if (Z_ISREF_P(p) && Z_REFCOUNT_P(p) == 1) {
ZVAL_COPY(p, Z_REFVAL_P(p));
} else {
Z_ADDREF_P(p);
}
}
1999-04-08 02:10:10 +08:00
}
ZEND_API void ZEND_FASTCALL zval_copy_ctor_func(zval *zvalue)
1999-04-08 02:10:10 +08:00
{
if (EXPECTED(Z_TYPE_P(zvalue) == IS_ARRAY)) {
ZVAL_ARR(zvalue, zend_array_dup(Z_ARRVAL_P(zvalue)));
2017-09-13 06:45:57 +08:00
} else if (EXPECTED(Z_TYPE_P(zvalue) == IS_STRING)) {
2017-12-07 20:01:23 +08:00
ZEND_ASSERT(!ZSTR_IS_INTERNED(Z_STR_P(zvalue)));
CHECK_ZVAL_STRING(Z_STR_P(zvalue));
2017-09-13 06:45:57 +08:00
ZVAL_NEW_STR(zvalue, zend_string_dup(Z_STR_P(zvalue), 0));
} else {
ZEND_UNREACHABLE();
1999-04-08 02:10:10 +08:00
}
}