Skip to content

Commit ce8ab5f

Browse files
committed
Fix GH-17736: Assertion failure zend_reference_destroy()
The cache slot for FETCH_OBJ_W in function `test` is primed with the class for C. The next call uses a simplexml instance and reuses the same cache slot. simplexml's get_property_ptr handler does not use the cache slot, so the old values remain in the cache slot. When `zend_handle_fetch_obj_flags` is called this is not guarded by a check for the class entry. So we end up using the prop_info from the property C::$a instead of the simplexml property. This patch adds a reset to the cache slots in the property address fetch code and also in the extensions with a non-standard reference handler. This keeps the run time cache consistent and avoids the issue without complicating the fast paths. Closes GH-17739.
1 parent 6bb56fe commit ce8ab5f

File tree

11 files changed

+37
-0
lines changed

11 files changed

+37
-0
lines changed

NEWS

+2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@ PHP NEWS
22
|||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
33
?? ??? ????, PHP 8.3.19
44

5+
- Treewide:
6+
. Fixed bug GH-17736 (Assertion failure zend_reference_destroy()). (nielsdos)
57

68
27 Feb 2025, PHP 8.3.18RC1
79

Zend/zend_execute.c

+3
Original file line numberDiff line numberDiff line change
@@ -3232,6 +3232,9 @@ static zend_always_inline void zend_fetch_property_address(zval *result, zval *c
32323232
return;
32333233
}
32343234
}
3235+
} else if (prop_op_type == IS_CONST) {
3236+
/* CE mismatch, make cache slot consistent */
3237+
cache_slot[0] = cache_slot[1] = cache_slot[2] = NULL;
32353238
}
32363239

32373240
/* Pointer on property callback is required */

ext/date/php_date.c

+1
Original file line numberDiff line numberDiff line change
@@ -4399,6 +4399,7 @@ static zval *date_interval_get_property_ptr_ptr(zend_object *object, zend_string
43994399
zend_string_equals_literal(name, "days") ||
44004400
zend_string_equals_literal(name, "invert") ) {
44014401
/* Fallback to read_property. */
4402+
cache_slot[0] = cache_slot[1] = cache_slot[2] = NULL;
44024403
ret = NULL;
44034404
} else {
44044405
ret = zend_std_get_property_ptr_ptr(object, name, type, cache_slot);

ext/dom/php_dom.c

+1
Original file line numberDiff line numberDiff line change
@@ -303,6 +303,7 @@ static zval *dom_get_property_ptr_ptr(zend_object *object, zend_string *name, in
303303
return zend_std_get_property_ptr_ptr(object, name, type, cache_slot);
304304
}
305305

306+
cache_slot[0] = cache_slot[1] = cache_slot[2] = NULL;
306307
return NULL;
307308
}
308309

ext/pdo/pdo_stmt.c

+1
Original file line numberDiff line numberDiff line change
@@ -2495,6 +2495,7 @@ static zval *pdo_row_get_property_ptr_ptr(zend_object *object, zend_string *name
24952495
ZEND_IGNORE_VALUE(type);
24962496
ZEND_IGNORE_VALUE(cache_slot);
24972497

2498+
cache_slot[0] = cache_slot[1] = cache_slot[2] = NULL;
24982499
return NULL;
24992500
}
25002501

ext/simplexml/simplexml.c

+2
Original file line numberDiff line numberDiff line change
@@ -635,6 +635,8 @@ static zval *sxe_property_get_adr(zend_object *object, zend_string *zname, int f
635635
SXE_ITER type;
636636
zval member;
637637

638+
cache_slot[0] = cache_slot[1] = cache_slot[2] = NULL;
639+
638640
sxe = php_sxe_fetch_object(object);
639641
GET_NODE(sxe, node);
640642
if (UNEXPECTED(!node)) {

ext/simplexml/tests/gh17736.phpt

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
--TEST--
2+
GH-17736 (Assertion failure zend_reference_destroy())
3+
--EXTENSIONS--
4+
simplexml
5+
--FILE--
6+
<?php
7+
$o1 = new SimpleXMLElement('<a/>');
8+
class C {
9+
public int $a = 1;
10+
}
11+
function test($obj) {
12+
$ref =& $obj->a;
13+
}
14+
$obj = new C;
15+
test($obj);
16+
test($o1);
17+
echo "Done\n";
18+
?>
19+
--EXPECT--
20+
Done

ext/snmp/snmp.c

+1
Original file line numberDiff line numberDiff line change
@@ -1861,6 +1861,7 @@ static zval *php_snmp_get_property_ptr_ptr(zend_object *object, zend_string *nam
18611861
return zend_std_get_property_ptr_ptr(object, name, type, cache_slot);
18621862
}
18631863

1864+
cache_slot[0] = cache_slot[1] = cache_slot[2] = NULL;
18641865
return NULL;
18651866
}
18661867

ext/spl/spl_array.c

+2
Original file line numberDiff line numberDiff line change
@@ -844,6 +844,8 @@ static zval *spl_array_get_property_ptr_ptr(zend_object *object, zend_string *na
844844

845845
if ((intern->ar_flags & SPL_ARRAY_ARRAY_AS_PROPS) != 0
846846
&& !zend_std_has_property(object, name, ZEND_PROPERTY_EXISTS, NULL)) {
847+
cache_slot[0] = cache_slot[1] = cache_slot[2] = NULL;
848+
847849
/* If object has offsetGet() overridden, then fallback to read_property,
848850
* which will call offsetGet(). */
849851
zval member;

ext/xmlreader/php_xmlreader.c

+2
Original file line numberDiff line numberDiff line change
@@ -121,6 +121,8 @@ zval *xmlreader_get_property_ptr_ptr(zend_object *object, zend_string *name, int
121121
zval *retval = NULL;
122122
xmlreader_prop_handler *hnd = NULL;
123123

124+
cache_slot[0] = cache_slot[1] = cache_slot[2] = NULL;
125+
124126
obj = php_xmlreader_fetch_object(object);
125127

126128
if (obj->prop_handler != NULL) {

ext/zip/php_zip.c

+2
Original file line numberDiff line numberDiff line change
@@ -890,6 +890,8 @@ static zval *php_zip_get_property_ptr_ptr(zend_object *object, zend_string *name
890890
zval *retval = NULL;
891891
zip_prop_handler *hnd = NULL;
892892

893+
cache_slot[0] = cache_slot[1] = cache_slot[2] = NULL;
894+
893895
obj = php_zip_fetch_object(object);
894896

895897
if (obj->prop_handler != NULL) {

0 commit comments

Comments
 (0)