1999-04-07 18:10:10 +00:00
|
|
|
/*
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| Zend Engine |
|
|
|
|
+----------------------------------------------------------------------+
|
2014-01-03 03:08:10 +00:00
|
|
|
| Copyright (c) 1998-2014 Zend Technologies Ltd. (http://www.zend.com) |
|
1999-04-07 18:10:10 +00:00
|
|
|
+----------------------------------------------------------------------+
|
2001-12-11 15:16:21 +00:00
|
|
|
| This source file is subject to version 2.00 of the Zend license, |
|
1999-07-16 14:58:16 +00:00
|
|
|
| that is bundled with this package in the file LICENSE, and is |
|
2003-06-10 20:04:29 +00:00
|
|
|
| available through the world-wide-web at the following url: |
|
2001-12-11 15:16:21 +00:00
|
|
|
| http://www.zend.com/license/2_00.txt. |
|
1999-07-16 14:58:16 +00: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-07 18:10:10 +00:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| Authors: Andi Gutmans <andi@zend.com> |
|
|
|
|
| Zeev Suraski <zeev@zend.com> |
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
*/
|
|
|
|
|
2003-02-01 01:49:15 +00:00
|
|
|
/* $Id$ */
|
1999-07-16 14:58:16 +00:00
|
|
|
|
1999-04-07 18:10:10 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include "zend.h"
|
|
|
|
#include "zend_API.h"
|
2013-10-31 07:57:12 +00:00
|
|
|
#include "zend_ast.h"
|
1999-04-07 18:10:10 +00:00
|
|
|
#include "zend_globals.h"
|
|
|
|
#include "zend_constants.h"
|
|
|
|
#include "zend_list.h"
|
|
|
|
|
2004-09-26 20:03:54 +00:00
|
|
|
ZEND_API void _zval_dtor_func(zval *zvalue ZEND_FILE_LINE_DC)
|
1999-04-07 18:10:10 +00:00
|
|
|
{
|
2007-09-28 19:52:53 +00:00
|
|
|
switch (Z_TYPE_P(zvalue) & IS_CONSTANT_TYPE_MASK) {
|
1999-04-07 18:10:10 +00:00
|
|
|
case IS_STRING:
|
|
|
|
case IS_CONSTANT:
|
2001-07-16 15:48:31 +00:00
|
|
|
CHECK_ZVAL_STRING_REL(zvalue);
|
2014-02-17 13:59:18 +00:00
|
|
|
STR_RELEASE(Z_STR_P(zvalue));
|
1999-04-07 18:10:10 +00:00
|
|
|
break;
|
2000-05-31 19:07:09 +00:00
|
|
|
case IS_ARRAY:
|
|
|
|
case IS_CONSTANT_ARRAY: {
|
2001-07-27 10:10:39 +00:00
|
|
|
TSRMLS_FETCH();
|
1999-04-07 18:10:10 +00:00
|
|
|
|
2014-02-12 14:08:11 +00:00
|
|
|
if (Z_ARRVAL_P(zvalue) != &EG(symbol_table).ht) {
|
2012-10-18 12:10:35 +00:00
|
|
|
/* break possible cycles */
|
|
|
|
Z_TYPE_P(zvalue) = IS_NULL;
|
2014-02-10 06:04:30 +00:00
|
|
|
zend_hash_destroy(Z_ARRVAL_P(zvalue));
|
2014-02-12 14:08:11 +00:00
|
|
|
efree(Z_ARR_P(zvalue));
|
|
|
|
}
|
1999-04-07 18:10:10 +00:00
|
|
|
}
|
|
|
|
break;
|
2013-10-31 07:57:12 +00:00
|
|
|
case IS_CONSTANT_AST:
|
2014-02-10 06:04:30 +00:00
|
|
|
zend_ast_destroy(Z_AST_P(zvalue)->ast);
|
2014-02-12 14:08:11 +00:00
|
|
|
efree(Z_AST_P(zvalue));
|
2013-10-31 07:57:12 +00:00
|
|
|
break;
|
1999-07-10 11:45:23 +00:00
|
|
|
case IS_OBJECT:
|
2002-03-15 16:26:17 +00:00
|
|
|
{
|
|
|
|
TSRMLS_FETCH();
|
|
|
|
|
2014-02-14 13:48:45 +00:00
|
|
|
OBJ_RELEASE(Z_OBJ_P(zvalue));
|
2002-03-15 16:26:17 +00:00
|
|
|
}
|
1999-04-07 18:10:10 +00:00
|
|
|
break;
|
2002-03-15 16:26:17 +00:00
|
|
|
case IS_RESOURCE:
|
|
|
|
{
|
2001-07-30 04:54:16 +00:00
|
|
|
TSRMLS_FETCH();
|
|
|
|
|
2014-02-10 06:04:30 +00:00
|
|
|
if (Z_DELREF_P(zvalue) == 0) {
|
|
|
|
/* destroy resource */
|
|
|
|
zend_list_delete(Z_RES_P(zvalue));
|
|
|
|
}
|
2001-07-30 04:54:16 +00:00
|
|
|
}
|
1999-04-07 18:10:10 +00:00
|
|
|
break;
|
2014-02-18 12:27:38 +00:00
|
|
|
case IS_REFERENCE:
|
|
|
|
if (Z_DELREF_P(zvalue) == 0) {
|
|
|
|
zval_dtor(Z_REFVAL_P(zvalue));
|
|
|
|
efree(Z_REF_P(zvalue));
|
|
|
|
}
|
|
|
|
break;
|
1999-04-07 18:10:10 +00:00
|
|
|
case IS_LONG:
|
|
|
|
case IS_DOUBLE:
|
|
|
|
case IS_BOOL:
|
2000-01-04 13:22:58 +00:00
|
|
|
case IS_NULL:
|
1999-04-07 18:10:10 +00:00
|
|
|
default:
|
2000-01-17 17:33:37 +00:00
|
|
|
return;
|
1999-04-07 18:10:10 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-17 13:59:18 +00:00
|
|
|
ZEND_API void _zval_dtor_func_for_ptr(zval *zvalue ZEND_FILE_LINE_DC)
|
|
|
|
{
|
|
|
|
switch (Z_TYPE_P(zvalue) & IS_CONSTANT_TYPE_MASK) {
|
|
|
|
case IS_STRING:
|
|
|
|
case IS_CONSTANT:
|
|
|
|
CHECK_ZVAL_STRING_REL(zvalue);
|
|
|
|
STR_FREE(Z_STR_P(zvalue));
|
|
|
|
break;
|
|
|
|
case IS_ARRAY:
|
|
|
|
case IS_CONSTANT_ARRAY: {
|
|
|
|
TSRMLS_FETCH();
|
|
|
|
|
|
|
|
if (Z_ARRVAL_P(zvalue) != &EG(symbol_table).ht) {
|
|
|
|
/* break possible cycles */
|
|
|
|
Z_TYPE_P(zvalue) = IS_NULL;
|
|
|
|
zend_hash_destroy(Z_ARRVAL_P(zvalue));
|
|
|
|
efree(Z_ARR_P(zvalue));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IS_CONSTANT_AST:
|
|
|
|
zend_ast_destroy(Z_AST_P(zvalue)->ast);
|
|
|
|
efree(Z_AST_P(zvalue));
|
|
|
|
break;
|
|
|
|
case IS_OBJECT:
|
|
|
|
{
|
|
|
|
TSRMLS_FETCH();
|
|
|
|
|
|
|
|
zend_objects_store_del(Z_OBJ_P(zvalue) TSRMLS_CC);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IS_RESOURCE:
|
|
|
|
{
|
|
|
|
TSRMLS_FETCH();
|
|
|
|
|
|
|
|
/* destroy resource */
|
|
|
|
zend_list_delete(Z_RES_P(zvalue));
|
|
|
|
}
|
|
|
|
break;
|
2014-02-18 12:27:38 +00:00
|
|
|
case IS_REFERENCE:
|
|
|
|
zval_dtor(Z_REFVAL_P(zvalue));
|
|
|
|
efree(Z_REF_P(zvalue));
|
|
|
|
break;
|
2014-02-17 13:59:18 +00:00
|
|
|
case IS_LONG:
|
|
|
|
case IS_DOUBLE:
|
|
|
|
case IS_BOOL:
|
|
|
|
case IS_NULL:
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
1999-04-07 18:10:10 +00:00
|
|
|
|
2003-08-24 17:32:47 +00:00
|
|
|
ZEND_API void _zval_internal_dtor(zval *zvalue ZEND_FILE_LINE_DC)
|
|
|
|
{
|
2007-09-28 19:52:53 +00:00
|
|
|
switch (Z_TYPE_P(zvalue) & IS_CONSTANT_TYPE_MASK) {
|
2003-08-24 17:32:47 +00:00
|
|
|
case IS_STRING:
|
|
|
|
case IS_CONSTANT:
|
|
|
|
CHECK_ZVAL_STRING_REL(zvalue);
|
2014-02-17 13:59:18 +00:00
|
|
|
STR_RELEASE(Z_STR_P(zvalue));
|
2003-08-24 17:32:47 +00:00
|
|
|
break;
|
|
|
|
case IS_ARRAY:
|
|
|
|
case IS_CONSTANT_ARRAY:
|
2013-11-06 18:21:07 +00:00
|
|
|
case IS_CONSTANT_AST:
|
2003-08-24 17:32:47 +00:00
|
|
|
case IS_OBJECT:
|
|
|
|
case IS_RESOURCE:
|
|
|
|
zend_error(E_CORE_ERROR, "Internal zval's can't be arrays, objects or resources");
|
|
|
|
break;
|
|
|
|
case IS_LONG:
|
|
|
|
case IS_DOUBLE:
|
|
|
|
case IS_BOOL:
|
|
|
|
case IS_NULL:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-17 13:59:18 +00:00
|
|
|
ZEND_API void _zval_internal_dtor_for_ptr(zval *zvalue ZEND_FILE_LINE_DC)
|
|
|
|
{
|
|
|
|
switch (Z_TYPE_P(zvalue) & IS_CONSTANT_TYPE_MASK) {
|
|
|
|
case IS_STRING:
|
|
|
|
case IS_CONSTANT:
|
|
|
|
CHECK_ZVAL_STRING_REL(zvalue);
|
|
|
|
STR_FREE(Z_STR_P(zvalue));
|
|
|
|
break;
|
|
|
|
case IS_ARRAY:
|
|
|
|
case IS_CONSTANT_ARRAY:
|
|
|
|
case IS_CONSTANT_AST:
|
|
|
|
case IS_OBJECT:
|
|
|
|
case IS_RESOURCE:
|
|
|
|
zend_error(E_CORE_ERROR, "Internal zval's can't be arrays, objects or resources");
|
|
|
|
break;
|
|
|
|
case IS_LONG:
|
|
|
|
case IS_DOUBLE:
|
|
|
|
case IS_BOOL:
|
|
|
|
case IS_NULL:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2003-08-24 17:32:47 +00:00
|
|
|
|
2014-02-10 06:04:30 +00:00
|
|
|
ZEND_API void zval_add_ref(zval *p)
|
1999-04-07 18:10:10 +00:00
|
|
|
{
|
2014-02-28 07:03:05 +00:00
|
|
|
if (Z_REFCOUNTED_P(p)) {
|
2014-03-06 08:37:46 +00:00
|
|
|
//???: autoconversion from reverence to ordinal value
|
|
|
|
if (Z_ISREF_P(p) && Z_REFCOUNT_P(p) == 1) {
|
|
|
|
ZVAL_DUP(p, Z_REFVAL_P(p));
|
|
|
|
} else {
|
|
|
|
Z_ADDREF_P(p);
|
|
|
|
}
|
2014-02-10 06:04:30 +00:00
|
|
|
}
|
1999-04-07 18:10:10 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 07:03:05 +00:00
|
|
|
ZEND_API void zval_add_ref_unref(zval *p)
|
|
|
|
{
|
|
|
|
if (Z_REFCOUNTED_P(p)) {
|
|
|
|
if (Z_ISREF_P(p)) {
|
|
|
|
if (Z_REFCOUNT_P(p) == 1) {
|
|
|
|
zval *q = Z_REFVAL_P(p);
|
|
|
|
ZVAL_DUP(p, q);
|
|
|
|
} else {
|
|
|
|
Z_ADDREF_P(p);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Z_ADDREF_P(p);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
1999-04-07 18:10:10 +00:00
|
|
|
|
2004-09-27 02:29:01 +00:00
|
|
|
ZEND_API void _zval_copy_ctor_func(zval *zvalue ZEND_FILE_LINE_DC)
|
1999-04-07 18:10:10 +00:00
|
|
|
{
|
2007-09-28 19:52:53 +00:00
|
|
|
switch (Z_TYPE_P(zvalue) & IS_CONSTANT_TYPE_MASK) {
|
2000-04-26 22:10:06 +00:00
|
|
|
case IS_CONSTANT:
|
1999-04-07 18:10:10 +00:00
|
|
|
case IS_STRING:
|
2001-07-16 15:48:31 +00:00
|
|
|
CHECK_ZVAL_STRING_REL(zvalue);
|
2014-02-18 21:12:05 +00:00
|
|
|
if (!IS_INTERNED(Z_STR_P(zvalue))) {
|
|
|
|
Z_STR_P(zvalue) = STR_DUP(Z_STR_P(zvalue), 0);
|
|
|
|
}
|
1999-04-07 18:10:10 +00:00
|
|
|
break;
|
2000-05-31 19:07:09 +00:00
|
|
|
case IS_ARRAY:
|
|
|
|
case IS_CONSTANT_ARRAY: {
|
2014-02-10 06:04:30 +00:00
|
|
|
HashTable *ht = Z_ARRVAL_P(zvalue);
|
2001-07-27 10:10:39 +00:00
|
|
|
TSRMLS_FETCH();
|
1999-04-07 18:10:10 +00:00
|
|
|
|
2014-02-10 06:04:30 +00:00
|
|
|
if (ht == &EG(symbol_table).ht) {
|
2004-09-27 02:29:01 +00:00
|
|
|
return; /* do nothing */
|
1999-04-07 18:10:10 +00:00
|
|
|
}
|
2014-02-10 06:04:30 +00:00
|
|
|
ZVAL_NEW_ARR(zvalue);
|
|
|
|
zend_hash_init(Z_ARRVAL_P(zvalue), zend_hash_num_elements(ht), NULL, ZVAL_PTR_DTOR, 0);
|
2014-02-17 13:59:18 +00:00
|
|
|
zend_hash_copy(Z_ARRVAL_P(zvalue), ht, zval_add_ref);
|
1999-04-07 18:10:10 +00:00
|
|
|
}
|
|
|
|
break;
|
2014-02-10 06:04:30 +00:00
|
|
|
case IS_CONSTANT_AST: {
|
|
|
|
zend_ast_ref *ast = emalloc(sizeof(zend_ast_ref));
|
|
|
|
|
|
|
|
ast->gc.refcount = 1;
|
|
|
|
ast->gc.u.v.type = IS_CONSTANT_AST;
|
|
|
|
ast->ast = zend_ast_copy(Z_ASTVAL_P(zvalue));
|
|
|
|
Z_AST_P(zvalue) = ast;
|
|
|
|
}
|
2013-10-31 07:57:12 +00:00
|
|
|
break;
|
2001-08-07 03:17:33 +00:00
|
|
|
case IS_OBJECT:
|
2014-02-10 06:04:30 +00:00
|
|
|
case IS_RESOURCE:
|
|
|
|
case IS_REFERENCE:
|
|
|
|
Z_ADDREF_P(zvalue);
|
1999-04-07 18:10:10 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ZEND_API int zend_print_variable(zval *var)
|
|
|
|
{
|
|
|
|
return zend_print_zval(var, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-01-17 17:33:37 +00:00
|
|
|
ZEND_API void _zval_dtor_wrapper(zval *zvalue)
|
1999-08-27 19:17:19 +00:00
|
|
|
{
|
2009-12-25 13:11:18 +00:00
|
|
|
TSRMLS_FETCH();
|
|
|
|
|
|
|
|
GC_REMOVE_ZVAL_FROM_BUFFER(zvalue);
|
2000-01-17 17:33:37 +00:00
|
|
|
zval_dtor(zvalue);
|
1999-08-27 19:17:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-11 10:07:10 +00:00
|
|
|
#if ZEND_DEBUG
|
|
|
|
ZEND_API void _zval_copy_ctor_wrapper(zval *zvalue)
|
|
|
|
{
|
|
|
|
zval_copy_ctor(zvalue);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-08-24 17:32:47 +00:00
|
|
|
ZEND_API void _zval_internal_dtor_wrapper(zval *zvalue)
|
|
|
|
{
|
|
|
|
zval_internal_dtor(zvalue);
|
|
|
|
}
|
|
|
|
|
1999-08-27 19:17:19 +00:00
|
|
|
|
2014-02-10 06:04:30 +00:00
|
|
|
ZEND_API void _zval_ptr_dtor_wrapper(zval *zval_ptr)
|
1999-08-27 19:17:19 +00:00
|
|
|
{
|
2000-01-17 18:45:46 +00:00
|
|
|
zval_ptr_dtor(zval_ptr);
|
1999-08-27 19:17:19 +00:00
|
|
|
}
|
2003-08-24 17:32:47 +00:00
|
|
|
|
2014-02-10 06:04:30 +00:00
|
|
|
ZEND_API void _zval_internal_ptr_dtor_wrapper(zval *zval_ptr)
|
2003-08-24 17:32:47 +00:00
|
|
|
{
|
|
|
|
zval_internal_ptr_dtor(zval_ptr);
|
|
|
|
}
|
1999-08-27 19:17:19 +00:00
|
|
|
#endif
|
|
|
|
|
2014-02-10 06:04:30 +00:00
|
|
|
ZEND_API int zval_copy_static_var(zval *p TSRMLS_DC, int num_args, va_list args, zend_hash_key *key) /* {{{ */
|
2010-06-08 15:56:36 +00:00
|
|
|
{
|
|
|
|
HashTable *target = va_arg(args, HashTable*);
|
|
|
|
zend_bool is_ref;
|
2014-02-10 06:04:30 +00:00
|
|
|
zval tmp;
|
2010-06-08 15:56:36 +00:00
|
|
|
|
2014-02-10 06:04:30 +00:00
|
|
|
if (Z_TYPE_P(p) & (IS_LEXICAL_VAR|IS_LEXICAL_REF)) {
|
|
|
|
is_ref = Z_TYPE_P(p) & IS_LEXICAL_REF;
|
2010-06-08 15:56:36 +00:00
|
|
|
|
|
|
|
if (!EG(active_symbol_table)) {
|
|
|
|
zend_rebuild_symbol_table(TSRMLS_C);
|
|
|
|
}
|
2014-02-10 06:04:30 +00:00
|
|
|
p = zend_hash_find(EG(active_symbol_table), key->key);
|
|
|
|
if (!p) {
|
|
|
|
p = &tmp;
|
|
|
|
ZVAL_NULL(&tmp);
|
|
|
|
if (is_ref) {
|
|
|
|
ZVAL_NEW_REF(&tmp, &tmp);
|
|
|
|
zend_hash_add(EG(active_symbol_table), key->key, &tmp);
|
2010-06-08 15:56:36 +00:00
|
|
|
} else {
|
2014-02-10 06:04:30 +00:00
|
|
|
zend_error(E_NOTICE,"Undefined variable: %s", key->key->val);
|
2010-06-08 15:56:36 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (is_ref) {
|
2014-02-12 14:08:11 +00:00
|
|
|
SEPARATE_ZVAL_TO_MAKE_IS_REF(p);
|
2014-02-19 12:50:09 +00:00
|
|
|
/*
|
2014-02-10 06:04:30 +00:00
|
|
|
if (!Z_ISREF_P(p)) {
|
2014-03-05 09:55:56 +00:00
|
|
|
if (Z_REFCOUNTED_P(p) && Z_REFCOUNT_P(p) > 1) {
|
2014-02-10 06:04:30 +00:00
|
|
|
Z_DELREF_P(p);
|
|
|
|
ZVAL_NEW_REF(p, p);
|
|
|
|
zval_copy_ctor(Z_REFVAL_P(p));
|
|
|
|
Z_SET_REFCOUNT_P(Z_REFVAL_P(p), 1);
|
|
|
|
} else {
|
|
|
|
ZVAL_NEW_REF(p, p);
|
|
|
|
}
|
|
|
|
}
|
2014-02-19 12:50:09 +00:00
|
|
|
*/
|
2014-02-10 06:04:30 +00:00
|
|
|
} else if (Z_ISREF_P(p)) {
|
2014-02-12 14:08:11 +00:00
|
|
|
ZVAL_COPY_VALUE(&tmp, Z_REFVAL_P(p));
|
2014-02-10 06:04:30 +00:00
|
|
|
if (Z_REFCOUNTED(tmp) && Z_REFCOUNT(tmp) > 1) {
|
|
|
|
zval_copy_ctor(&tmp);
|
|
|
|
Z_SET_REFCOUNT(tmp, 0);
|
|
|
|
}
|
|
|
|
p = &tmp;
|
2010-06-08 15:56:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-02-10 06:04:30 +00:00
|
|
|
if (zend_hash_add(target, key->key, p)) {
|
|
|
|
if (Z_REFCOUNTED_P(p)) {
|
|
|
|
Z_ADDREF_P(p);
|
|
|
|
}
|
2010-06-08 15:56:36 +00:00
|
|
|
}
|
|
|
|
return ZEND_HASH_APPLY_KEEP;
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
1999-04-07 18:10:10 +00:00
|
|
|
/*
|
|
|
|
* Local variables:
|
|
|
|
* tab-width: 4
|
|
|
|
* c-basic-offset: 4
|
2003-02-01 01:49:15 +00:00
|
|
|
* indent-tabs-mode: t
|
1999-04-07 18:10:10 +00:00
|
|
|
* End:
|
|
|
|
*/
|