Bug #70262 (Accessing array crashes)
--FILE--
<?php
-class C {
- public $arguments;
- public function __construct($arg) {
- $this->arguments = $arg;
- }
-}
-function & a(&$arg) {
- $c = new C($arg);
- $arg[] = $c;
- return $c;
-}
+$array = array();
+$array[] = 1; // make this not immutable
-function c($arr) {
- a($arr)->arguments[0] = "bad";
+$extra = $array; // make the refcount == 2
+
+class A {
+ public function getObj($array) {
+ $obj = new Stdclass;
+ $obj->arr = $array; // make the refcount == 3;
+ return $obj;
+ }
}
-$arr = array();
-$arr[] = "foo";
-$arr[] = "bar";
-c($arr);
-var_dump($arr);
+$a = new A;
+$a->getObj($array) //use function call to get a refcount == 1 IS_VAR object
+ ->arr // FETCH_OBJ_W will EXTRACT_ZVAL_PTR because getObj() result a refcount == 1 object (READY_TO_DESTROY)
+ [0] = "test"; //We will get a refcount == 3 array (not a IS_INDIRCT) in ZEND_ASSIGN_DIM_SPEC_VAR_CONST_HANDLER
+
+var_dump($array);
?>
--EXPECT--
-array(2) {
+array(1) {
[0]=>
- string(3) "foo"
- [1]=>
- string(3) "bar"
+ int(1)
}
if (EXPECTED(Z_TYPE_P(ret) == IS_INDIRECT)) {
*should_free = NULL;
ret = Z_INDIRECT_P(ret);
- } else if (!Z_REFCOUNTED_P(ret)) {
- *should_free = ret; /* immutable array may be converted to regular */
- } else if (Z_REFCOUNT_P(ret) == 1) {
- *should_free = ret;
} else {
- *should_free = NULL;
- Z_DELREF_P(ret);
+ *should_free = ret; /* immutable array may be converted to regular */
}
return ret;
}
zval *var_ptr;
var_ptr = EX_VAR(opline->op1.var);
- if (UNEXPECTED(Z_ISREF_P(var_ptr))) {
- if (UNEXPECTED(Z_REFCOUNT_P(var_ptr) == 1)) {
- ZVAL_UNREF(var_ptr);
- }
- } else if (Z_COPYABLE_P(var_ptr) && Z_REFCOUNT_P(var_ptr) > 1) {
+ if (Z_TYPE_P(var_ptr) != IS_OBJECT &&
+ !Z_ISREF_P(var_ptr) &&
+ Z_REFCOUNTED_P(var_ptr) &&
+ Z_REFCOUNT_P(var_ptr) > 1) {
+
Z_DELREF_P(var_ptr);
ZVAL_DUP(EX_VAR(opline->op1.var), var_ptr);
}
zval *var_ptr;
var_ptr = EX_VAR(opline->op1.var);
- if (UNEXPECTED(Z_ISREF_P(var_ptr))) {
- if (UNEXPECTED(Z_REFCOUNT_P(var_ptr) == 1)) {
- ZVAL_UNREF(var_ptr);
- }
- } else if (Z_COPYABLE_P(var_ptr) && Z_REFCOUNT_P(var_ptr) > 1) {
+ if (Z_TYPE_P(var_ptr) != IS_OBJECT &&
+ !Z_ISREF_P(var_ptr) &&
+ Z_REFCOUNTED_P(var_ptr) &&
+ Z_REFCOUNT_P(var_ptr) > 1) {
+
Z_DELREF_P(var_ptr);
ZVAL_DUP(EX_VAR(opline->op1.var), var_ptr);
}