Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix zend_get_property_info_for_slot() for lazy objects #15855

Merged
merged 1 commit into from
Sep 16, 2024

Conversation

arnaud-lb
Copy link
Member

@arnaud-lb arnaud-lb commented Sep 12, 2024

zend_get_property_info_for_slot(obj, slot) assumes that 'slot' belongs to 'obj', but that may not be the case for lazy proxies.

Fortunatley, the property info is often already available in path when it is nedded.

For other cases, I make zend_get_property_info_for_slot() aware of lazy objects, and add zend_get_property_info_for_slot_self() for cases where the 'slot' is known to belong to the object itself.

Fixes oss-fuzz #71446

VM uses zend_get_property_info_for_slot() for variable property accesses such as $obj->$prop. I retrieve the property info by inspecting the cache_slot, as it's set by the read_property or get_property_ptr_ptr handlers. An alternative would be to add a zend_property_info ** parameter to these handlers.

zend_get_property_info_for_slot(obj, slot) assumes that 'slot' belongs to 'obj',
but that may not be the case for lazy proxies.

Fortunatley, the property info is often already available in path when it is
nedded.

For other cases, I make zend_get_property_info_for_slot() aware of lazy objects,
and add zend_get_property_info_for_slot_self() for cases where the 'slot' is
known to belong to the object itself.

Fixes oss-fuzz #71446
Copy link
Member

@dstogov dstogov left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't understand this in all details, but I think it's OK.

Comment on lines -1052 to +1053
cache_slot = (OP2_TYPE == IS_CONST) ? CACHE_ADDR((opline+1)->extended_value) : NULL;
cache_slot = (OP2_TYPE == IS_CONST) ? CACHE_ADDR((opline+1)->extended_value) : _cache_slot;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

How does this change the cost for non CONST op2?
On one hand now get_property_ptr_ptr has to make more work, on the other hand zend_object_fetch_property_type_info is eliminated.
Can you measure the difference, please.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

With this micro bench under callgrind:

<?php

class C {
    public int $a = 0;
}

$obj = new C();
$a = 'a';
for ($i = 0; $i < 1_000_000; $i++) {
    $obj->$a += 2;
}
base:  mean:  264465884.0000;  stddev:  0.0000;  diff:  +0.00%
pr:    mean:  250464574.0000;  stddev:  0.0000;  diff:  -5.29%
alt:   mean:  246467362.0000;  stddev:  0.0000;  diff:  -6.81%

pr is this branch, and alt is an alternative in which I added a zend_property_info **prop_info_p parameter to zend_object_read_property_t and zend_object_get_property_ptr_ptr_t.

alt is faster than pr for op2 != CONST, however it's slower for op2 = CONST:

<?php

class C {
    public int $a = 0;
}

$obj = new C();
for ($i = 0; $i < 1_000_000; $i++) {
    $obj->a += 2;
}
base:  mean:  192463206.0000;  stddev:  0.0000;  diff:  +0.00%
pr:    mean:  192461896.0000;  stddev:  0.0000;  diff:  -0.00%
alt:   mean:  196464687.0000;  stddev:  0.0000;  diff:  +2.08%

@arnaud-lb arnaud-lb merged commit c65e042 into php:master Sep 16, 2024
10 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants