--- /dev/null
+--TEST--
+Covariant return-by-ref constraints
+--FILE--
+<?php
+
+class A implements ArrayAccess {
+ public $foo = array();
+
+ public function &offsetGet($n) {
+ return $this->foo[$n];
+ }
+
+ public function offsetSet($n, $v) {
+ }
+ public function offsetUnset($n) {
+ }
+ public function offsetExists($n) {
+ }
+}
+
+$a = new A;
+
+$a['foo']['bar'] = 2;
+
+var_dump($a);
+
+?>
+==DONE==
+--EXPECTF--
+object(A)#1 (1) {
+ ["foo"]=>
+ array(1) {
+ ["foo"]=>
+ array(1) {
+ ["bar"]=>
+ int(2)
+ }
+ }
+}
+==DONE==
return 0;
}
- if ((fe->common.fn_flags & ZEND_ACC_RETURN_REFERENCE) !=
- (proto->common.fn_flags & ZEND_ACC_RETURN_REFERENCE)) {
+ /* by-ref constraints on return values are covariant */
+ if ((proto->common.fn_flags & ZEND_ACC_RETURN_REFERENCE)
+ && !(fe->common.fn_flags & ZEND_ACC_RETURN_REFERENCE)) {
return 0;
}
/* Incompatible type hint */
return 0;
}
+
+ /* by-ref constraints on arguments are invariant */
if (fe->common.arg_info[i].pass_by_reference != proto->common.arg_info[i].pass_by_reference) {
return 0;
}