From 8a442a33c319a9663212e732dda4cd717d4efff6 Mon Sep 17 00:00:00 2001 From: Nikita Popov Date: Fri, 29 Jul 2016 19:07:04 +0200 Subject: Fix ref handling in array_merge/replace_recursive() As usual, if the reference has rc=1 it really isn't a reference. --- ext/standard/array.c | 26 +++++---------- .../array/array_merge_replace_recursive_refs.phpt | 38 ++++++++++++++++++++++ 2 files changed, 46 insertions(+), 18 deletions(-) create mode 100644 ext/standard/tests/array/array_merge_replace_recursive_refs.phpt diff --git a/ext/standard/array.c b/ext/standard/array.c index e5c38f2906..538460bdcc 100644 --- a/ext/standard/array.c +++ b/ext/standard/array.c @@ -3075,16 +3075,12 @@ PHPAPI int php_array_merge_recursive(HashTable *dest, HashTable *src) /* {{{ */ } zval_ptr_dtor(&tmp); } else { - if (Z_REFCOUNTED_P(src_entry)) { - Z_ADDREF_P(src_entry); - } - zend_hash_add_new(dest, string_key, src_entry); + zval *zv = zend_hash_add_new(dest, string_key, src_entry); + zval_add_ref(zv); } } else { - if (Z_REFCOUNTED_P(src_entry)) { - Z_ADDREF_P(src_entry); - } - zend_hash_next_index_insert_new(dest, src_entry); + zval *zv = zend_hash_next_index_insert_new(dest, src_entry); + zval_add_ref(zv); } } ZEND_HASH_FOREACH_END(); return 1; @@ -3134,11 +3130,8 @@ PHPAPI int php_array_replace_recursive(HashTable *dest, HashTable *src) /* {{{ * (Z_TYPE_P(dest_entry) != IS_ARRAY && (!Z_ISREF_P(dest_entry) || Z_TYPE_P(Z_REFVAL_P(dest_entry)) != IS_ARRAY))) { - if (Z_REFCOUNTED_P(src_entry)) { - Z_ADDREF_P(src_entry); - } - zend_hash_update(dest, string_key, src_entry); - + zval *zv = zend_hash_update(dest, string_key, src_entry); + zval_add_ref(zv); continue; } } else { @@ -3147,11 +3140,8 @@ PHPAPI int php_array_replace_recursive(HashTable *dest, HashTable *src) /* {{{ * (Z_TYPE_P(dest_entry) != IS_ARRAY && (!Z_ISREF_P(dest_entry) || Z_TYPE_P(Z_REFVAL_P(dest_entry)) != IS_ARRAY))) { - if (Z_REFCOUNTED_P(src_entry)) { - Z_ADDREF_P(src_entry); - } - zend_hash_index_update(dest, num_key, src_entry); - + zval *zv = zend_hash_index_update(dest, num_key, src_entry); + zval_add_ref(zv); continue; } } diff --git a/ext/standard/tests/array/array_merge_replace_recursive_refs.phpt b/ext/standard/tests/array/array_merge_replace_recursive_refs.phpt new file mode 100644 index 0000000000..ac8be2ba2f --- /dev/null +++ b/ext/standard/tests/array/array_merge_replace_recursive_refs.phpt @@ -0,0 +1,38 @@ +--TEST-- +array_merge/replace_recursive() should unwrap references with rc=1 +--FILE-- + +--EXPECT-- +array(1) { + [0]=> + array(1) { + [0]=> + int(24) + } +} +array(2) { + [0]=> + int(42) + [1]=> + int(24) +} -- cgit v1.2.1