summaryrefslogtreecommitdiff
path: root/Zend
diff options
context:
space:
mode:
authorZeev Suraski <zeev@php.net>2003-03-18 16:30:23 +0000
committerZeev Suraski <zeev@php.net>2003-03-18 16:30:23 +0000
commit40326f6adf7110bd6676bf7445cff7483cf173fb (patch)
treed4329286d2fd1817ff7892bb263713ced31a64b7 /Zend
parentb21fdb762a7d98241b8070e3b22a7310cfb014da (diff)
downloadphp-git-40326f6adf7110bd6676bf7445cff7483cf173fb.tar.gz
- Fix situation where a derived class declares a public (or implicit public)
with the same name as a private in the parent - Optimize 'static binding' of private properties a bit
Diffstat (limited to 'Zend')
-rw-r--r--Zend/zend_compile.c3
-rw-r--r--Zend/zend_object_handlers.c38
2 files changed, 30 insertions, 11 deletions
diff --git a/Zend/zend_compile.c b/Zend/zend_compile.c
index 749d8f4935..a86a565ed0 100644
--- a/Zend/zend_compile.c
+++ b/Zend/zend_compile.c
@@ -1682,6 +1682,9 @@ static zend_bool do_inherit_property_access_check(HashTable *target_ht, zend_pro
zend_class_entry *parent_ce = ce->parent;
if (parent_info->flags & ZEND_ACC_PRIVATE) {
+ if (zend_hash_quick_find(&ce->properties_info, hash_key->arKey, hash_key->nKeyLength, hash_key->h, (void **) &child_info)==SUCCESS) {
+ child_info->flags |= ZEND_ACC_CHANGED;
+ }
return 0; /* don't copy access information to child */
}
diff --git a/Zend/zend_object_handlers.c b/Zend/zend_object_handlers.c
index 7db7124ea3..8c193afbf9 100644
--- a/Zend/zend_object_handlers.c
+++ b/Zend/zend_object_handlers.c
@@ -179,24 +179,40 @@ inline int zend_verify_property_access(zend_property_info *property_info, zend_c
static inline zend_property_info *zend_get_property_info(zend_object *zobj, zval *member TSRMLS_DC)
{
- zend_property_info *property_info;
- zend_bool found_info_in_object = 0;
+ zend_property_info *property_info = NULL;
+ zend_property_info *scope_property_info;
+ zend_bool denied_access = 0;
+
ulong h = zend_get_hash_value(Z_STRVAL_P(member), Z_STRLEN_P(member)+1);
if (zend_hash_quick_find(&zobj->ce->properties_info, Z_STRVAL_P(member), Z_STRLEN_P(member)+1, h, (void **) &property_info)==SUCCESS) {
if (zend_verify_property_access(property_info, zobj->ce TSRMLS_CC)) {
- return property_info;
+ if (property_info->flags & ZEND_ACC_CHANGED
+ && !(property_info->flags & ZEND_ACC_PRIVATE)) {
+ /* We still need to make sure that we're not in a context
+ * where the right property is a different 'statically linked' private
+ * continue checking below...
+ */
+ } else {
+ return property_info;
+ }
} else {
- found_info_in_object = 1;
+ /* Try to look in the scope instead */
+ denied_access = 1;
}
}
- if (EG(scope)
- && zend_hash_quick_find(&EG(scope)->properties_info, Z_STRVAL_P(member), Z_STRLEN_P(member)+1, h, (void **) &property_info)==SUCCESS
- && property_info->flags & ZEND_ACC_PRIVATE) {
- /* ok */
- } else if (found_info_in_object) {
- /* Information was available, but we were denied access. Error out. */
- zend_error(E_ERROR, "Cannot access %s property %s::$%s", zend_visibility_string(property_info->flags), zobj->ce->name, Z_STRVAL_P(member));
+ if (EG(scope) != zobj->ce
+ && EG(scope)
+ && zend_hash_quick_find(&EG(scope)->properties_info, Z_STRVAL_P(member), Z_STRLEN_P(member)+1, h, (void **) &scope_property_info)==SUCCESS
+ && scope_property_info->flags & ZEND_ACC_PRIVATE) {
+ return scope_property_info;
+ } else if (property_info) {
+ if (denied_access) {
+ /* Information was available, but we were denied access. Error out. */
+ zend_error(E_ERROR, "Cannot access %s property %s::$%s", zend_visibility_string(property_info->flags), zobj->ce->name, Z_STRVAL_P(member));
+ } else {
+ /* fall through, return property_info... */
+ }
} else {
EG(std_property_info).flags = ZEND_ACC_PUBLIC;
EG(std_property_info).name = Z_STRVAL_P(member);