summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPedro Magalhães <mail@pmmaga.net>2017-01-26 23:50:56 +0100
committerJoe Watkins <krakjoe@php.net>2017-01-28 06:28:10 +0000
commitb67eb3440bb244adf6957bf2c68aeeaa6efc8c8d (patch)
tree297c7842cd672aabd023510186cf23c9684b1279
parentbf914ff75a79aaacffebf689e6dd2b17175070b0 (diff)
downloadphp-git-b67eb3440bb244adf6957bf2c68aeeaa6efc8c8d.tar.gz
Inheritance checks should not ignore parents if these implement an interface
-rw-r--r--Zend/tests/bug62358.phpt2
-rw-r--r--Zend/tests/bug73987.phpt18
-rw-r--r--Zend/tests/bug73987_1.phpt18
-rw-r--r--Zend/zend_inheritance.c9
4 files changed, 41 insertions, 6 deletions
diff --git a/Zend/tests/bug62358.phpt b/Zend/tests/bug62358.phpt
index 35bbc33835..8509383d46 100644
--- a/Zend/tests/bug62358.phpt
+++ b/Zend/tests/bug62358.phpt
@@ -23,4 +23,4 @@ class B extends A {
}
?>
--EXPECTF--
-Fatal error: Declaration of B::foo($var) must be compatible with I::foo() in %sbug62358.php on line 17
+Fatal error: Declaration of B::foo($var) must be compatible with A::foo() in %sbug62358.php on line 17
diff --git a/Zend/tests/bug73987.phpt b/Zend/tests/bug73987.phpt
new file mode 100644
index 0000000000..551565650c
--- /dev/null
+++ b/Zend/tests/bug73987.phpt
@@ -0,0 +1,18 @@
+--TEST--
+Bug #73987 (Method compatibility check looks to original definition and not parent)
+--FILE--
+<?php
+
+interface I {
+ public function example($a, $b, $c);
+}
+class A implements I {
+ public function example($a, $b = null, $c = null) { } // compatible with I::example
+}
+class B extends A {
+ public function example($a, $b, $c = null) { } // compatible with I::example
+}
+
+?>
+--EXPECTF--
+Fatal error: Declaration of B::example($a, $b, $c = NULL) must be compatible with A::example($a, $b = NULL, $c = NULL) in %s
diff --git a/Zend/tests/bug73987_1.phpt b/Zend/tests/bug73987_1.phpt
new file mode 100644
index 0000000000..6a0a157f29
--- /dev/null
+++ b/Zend/tests/bug73987_1.phpt
@@ -0,0 +1,18 @@
+--TEST--
+Bug #73987 (Method compatibility check looks to original definition and not parent)
+--FILE--
+<?php
+
+interface I {
+ public function example();
+}
+class A implements I {
+ public function example(): int { } // compatible with I::example
+}
+class B extends A {
+ public function example(): string { } // compatible with I::example
+}
+
+?>
+--EXPECTF--
+Fatal error: Declaration of B::example(): string must be compatible with A::example(): int in %s
diff --git a/Zend/zend_inheritance.c b/Zend/zend_inheritance.c
index fd1345f844..8ad5cc2e01 100644
--- a/Zend/zend_inheritance.c
+++ b/Zend/zend_inheritance.c
@@ -607,13 +607,12 @@ static void do_inheritance_check_on_method(zend_function *child, zend_function *
} else if (!(parent->common.fn_flags & ZEND_ACC_CTOR) || (parent->common.prototype && (parent->common.prototype->common.scope->ce_flags & ZEND_ACC_INTERFACE))) {
/* ctors only have a prototype if it comes from an interface */
child->common.prototype = parent->common.prototype ? parent->common.prototype : parent;
+ /* and if that is the case, we want to check inheritance against it */
+ if (parent->common.fn_flags & ZEND_ACC_CTOR) {
+ parent = child->common.prototype;
+ }
}
- if (child->common.prototype && (
- child->common.prototype->common.fn_flags & ZEND_ACC_ABSTRACT
- )) {
- parent = child->common.prototype;
- }
if (UNEXPECTED(!zend_do_perform_implementation_check(child, parent))) {
int error_level;
const char *error_verb;