summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorNikita Popov <nikita.ppv@gmail.com>2019-02-12 17:37:34 +0100
committerNikita Popov <nikita.ppv@gmail.com>2019-02-12 17:37:34 +0100
commit8afa2c632b872bb5c34641d4f67d8543cb82ea0e (patch)
tree0b0fa0455f6f9e1446ca22edfdefb5e64cfc84f6
parente683c189f2e4cb8f6f2904695738418eb1798ad3 (diff)
parent93641539429c41a8997016ee4a4502882b0a722e (diff)
downloadphp-git-8afa2c632b872bb5c34641d4f67d8543cb82ea0e.tar.gz
Merge branch 'PHP-7.4'
-rw-r--r--Zend/tests/bug76430.phpt22
-rw-r--r--Zend/tests/nested_method_and_function.phpt2
-rw-r--r--Zend/zend_compile.c28
3 files changed, 43 insertions, 9 deletions
diff --git a/Zend/tests/bug76430.phpt b/Zend/tests/bug76430.phpt
new file mode 100644
index 0000000000..f122d00984
--- /dev/null
+++ b/Zend/tests/bug76430.phpt
@@ -0,0 +1,22 @@
+--TEST--
+Bug #76430: __METHOD__ inconsistent outside of method
+--FILE--
+<?php
+
+class Foo {
+ const X = __METHOD__;
+}
+function foo() {
+ class Bar {
+ const X = __METHOD__;
+ }
+}
+
+foo();
+var_dump(Foo::X);
+var_dump(Bar::X);
+
+?>
+--EXPECT--
+string(0) ""
+string(0) ""
diff --git a/Zend/tests/nested_method_and_function.phpt b/Zend/tests/nested_method_and_function.phpt
index c63c601eaf..3d19a06088 100644
--- a/Zend/tests/nested_method_and_function.phpt
+++ b/Zend/tests/nested_method_and_function.phpt
@@ -30,7 +30,7 @@ $c();
--EXPECT--
string(7) "Baz\foo"
string(7) "Baz\foo"
-string(7) "Baz\Foo"
+string(0) ""
string(3) "bar"
string(12) "Baz\Foo::bar"
string(7) "Baz\Foo"
diff --git a/Zend/zend_compile.c b/Zend/zend_compile.c
index a3f5f07e57..1ebc7c1bf3 100644
--- a/Zend/zend_compile.c
+++ b/Zend/zend_compile.c
@@ -5737,6 +5737,7 @@ void zend_compile_func_decl(znode *result, zend_ast *ast, zend_bool toplevel) /*
zend_ast *return_type_ast = decl->child[3];
zend_bool is_method = decl->kind == ZEND_AST_METHOD;
+ zend_class_entry *orig_class_entry = CG(active_class_entry);
zend_op_array *orig_op_array = CG(active_op_array);
zend_op_array *op_array = zend_arena_alloc(&CG(arena), sizeof(zend_op_array));
zend_oparray_context orig_oparray_context;
@@ -5769,6 +5770,13 @@ void zend_compile_func_decl(znode *result, zend_ast *ast, zend_bool toplevel) /*
CG(active_op_array) = op_array;
+ /* Do not leak the class scope into free standing functions, even if they are dynamically
+ * defined inside a class method. This is necessary for correct handling of magic constants.
+ * For example __CLASS__ should always be "" inside a free standing function. */
+ if (decl->kind == ZEND_AST_FUNC_DECL) {
+ CG(active_class_entry) = NULL;
+ }
+
if (toplevel) {
op_array->fn_flags |= ZEND_ACC_TOP_LEVEL;
}
@@ -5816,6 +5824,7 @@ void zend_compile_func_decl(znode *result, zend_ast *ast, zend_bool toplevel) /*
zend_stack_del_top(&CG(loop_var_stack));
CG(active_op_array) = orig_op_array;
+ CG(active_class_entry) = orig_class_entry;
}
/* }}} */
@@ -6672,17 +6681,20 @@ static zend_bool zend_try_ct_eval_magic_const(zval *zv, zend_ast *ast) /* {{{ */
}
break;
case T_METHOD_C:
- if ((op_array && !op_array->scope && op_array->function_name) || (op_array->fn_flags & ZEND_ACC_CLOSURE)) {
- ZVAL_STR_COPY(zv, op_array->function_name);
- } else if (ce) {
- if (op_array && op_array->function_name) {
- ZVAL_NEW_STR(zv, zend_concat3(ZSTR_VAL(ce->name), ZSTR_LEN(ce->name), "::", 2,
+ /* Detect whether we are directly inside a class (e.g. a class constant) and treat
+ * this as not being inside a function. */
+ if (op_array && ce && !op_array->scope && !(op_array->fn_flags & ZEND_ACC_CLOSURE)) {
+ op_array = NULL;
+ }
+ if (op_array && op_array->function_name) {
+ if (op_array->scope) {
+ ZVAL_NEW_STR(zv, zend_concat3(
+ ZSTR_VAL(op_array->scope->name), ZSTR_LEN(op_array->scope->name),
+ "::", 2,
ZSTR_VAL(op_array->function_name), ZSTR_LEN(op_array->function_name)));
} else {
- ZVAL_STR_COPY(zv, ce->name);
+ ZVAL_STR_COPY(zv, op_array->function_name);
}
- } else if (op_array && op_array->function_name) {
- ZVAL_STR_COPY(zv, op_array->function_name);
} else {
ZVAL_EMPTY_STRING(zv);
}