-
Notifications
You must be signed in to change notification settings - Fork 7.8k
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 GH-17715: Handle preloaded internal function runtime cache #17835
base: PHP-8.4
Are you sure you want to change the base?
Conversation
This solely affects the builtin enum functions currently. Given that these are stored in SHM, we cannot simply hardwire a pointer into the internal function runtime cache on NTS too, but have to use a MAP_PTR (like on ZTS). Now, by design, the runtime cache of internal functions no longer is reset between requests, hence we need to store them explicitly as static runtime cache. On NTS builds we cannot trivially move the pointers into CG(internal_run_time_cache) as they're directly stored on the individual functions (on ZTS we could simply iterate the static map_ptrs). Hence, we have the choice between having opcache managing the internal run_time_cache for its preloaded functions itself or realloc CG(internal_run_time_cache) and iterate through all functions to assign the new address. We choose the latter for simplicity and initial speed. Note: map_ptr_static_last has been added as last element to zend_accel_shared_globals, so that accesses to it are compatible. We do not have to care about the ABI of creating new zend_accel_shared_globals structs. That's opcaches prerogative.
c37d9f8
to
ec35e3a
Compare
I didn't understand the problem from the PR name and comments.
I can't understand this.
I don't know about this design. @iluuu1994 this is related to PHP Enums. Please also take a look. |
@@ -61,7 +63,7 @@ PHP NEWS | |||
. Fix memory leak on overflow in _php_stream_scandir(). (nielsdos) | |||
|
|||
- Windows: | |||
. Fixed phpize for Windows 11 (24H2). (bwoebi) | |||
. Fixed phpize for Windows 11 (24H2). (Bob) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Noted. :)
op_arrays generated by preloaded scripts are stored in shm. Thus we need to use MAP_PTRs for NTS too. That's it. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is hard to understand (especially after a day of work at my job).
It's more complexity over the original PR that introduced the issue. Is this worth the cost to just make observers faster? I suppose the original PR can't be reverted at this point due to ABI constraints?
Signed-off-by: Bob Weinand <[email protected]>
4f8cf6e
to
7ce04d3
Compare
This was quite worth the cost, yes :-( I did have that assumption that all persistent internal functions were loaded before post_startup. Evidently preloaded enums violate this assumption... |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I understand the concept better now, but I still find it too complex to understand how this will interact with all of the details and possible implicit assumptions made in other parts of PHP.
I ran your test under Valgrind and I got the following Valgrind errors:
==226054== Invalid write of size 8
==226054== at 0x9EF417: zend_observer_fcall_install (zend_observer.c:126)
==226054== by 0x9EFAF4: zend_observer_fcall_begin_prechecked (zend_observer.c:262)
==226054== by 0x8E3E8B: zend_observer_fcall_begin_specialized (zend_observer.h:116)
==226054== by 0x8F7BC0: ZEND_DO_FCALL_SPEC_OBSERVER_HANDLER (zend_vm_execute.h:2141)
==226054== by 0x976552: execute_ex (zend_vm_execute.h:58859)
==226054== by 0x97AED8: zend_execute (zend_vm_execute.h:64236)
==226054== by 0xA1B97E: zend_execute_script (zend.c:1934)
==226054== by 0x7CA094: php_execute_script_ex (main.c:2575)
==226054== by 0x7CA212: php_execute_script (main.c:2615)
==226054== by 0xA1DE38: do_cli (php_cli.c:935)
==226054== by 0xA1EB86: main (php_cli.c:1310)
==226054== Address 0x7b0f5d8 is 0 bytes after a block of size 8 alloc'd
==226054== at 0x48457A8: malloc (vg_replace_malloc.c:446)
==226054== by 0x8779E9: __zend_malloc (zend_alloc.c:3280)
==226054== by 0x787C911: preload_load (ZendAccelerator.c:4388)
==226054== by 0x787D3E3: accel_preload (ZendAccelerator.c:4596)
==226054== by 0x787D94F: accel_finish_startup_preload (ZendAccelerator.c:4717)
==226054== by 0x787DDFC: accel_finish_startup (ZendAccelerator.c:4862)
==226054== by 0x7878771: accel_post_startup (ZendAccelerator.c:3373)
==226054== by 0xA19647: zend_post_startup (zend.c:1105)
==226054== by 0x7C99DE: php_module_startup (main.c:2324)
==226054== by 0xA1CD4E: php_cli_startup (php_cli.c:397)
==226054== by 0xA1EAEA: main (php_cli.c:1277)
==226054==
==226054== Invalid write of size 8
==226054== at 0x9EF48C: zend_observer_fcall_install (zend_observer.c:138)
==226054== by 0x9EFAF4: zend_observer_fcall_begin_prechecked (zend_observer.c:262)
==226054== by 0x8E3E8B: zend_observer_fcall_begin_specialized (zend_observer.h:116)
==226054== by 0x8F7BC0: ZEND_DO_FCALL_SPEC_OBSERVER_HANDLER (zend_vm_execute.h:2141)
==226054== by 0x976552: execute_ex (zend_vm_execute.h:58859)
==226054== by 0x97AED8: zend_execute (zend_vm_execute.h:64236)
==226054== by 0xA1B97E: zend_execute_script (zend.c:1934)
==226054== by 0x7CA094: php_execute_script_ex (main.c:2575)
==226054== by 0x7CA212: php_execute_script (main.c:2615)
==226054== by 0xA1DE38: do_cli (php_cli.c:935)
==226054== by 0xA1EB86: main (php_cli.c:1310)
==226054== Address 0x7b0f5d8 is 0 bytes after a block of size 8 alloc'd
==226054== at 0x48457A8: malloc (vg_replace_malloc.c:446)
==226054== by 0x8779E9: __zend_malloc (zend_alloc.c:3280)
==226054== by 0x787C911: preload_load (ZendAccelerator.c:4388)
==226054== by 0x787D3E3: accel_preload (ZendAccelerator.c:4596)
==226054== by 0x787D94F: accel_finish_startup_preload (ZendAccelerator.c:4717)
==226054== by 0x787DDFC: accel_finish_startup (ZendAccelerator.c:4862)
==226054== by 0x7878771: accel_post_startup (ZendAccelerator.c:3373)
==226054== by 0xA19647: zend_post_startup (zend.c:1105)
==226054== by 0x7C99DE: php_module_startup (main.c:2324)
==226054== by 0xA1CD4E: php_cli_startup (php_cli.c:397)
==226054== by 0xA1EAEA: main (php_cli.c:1277)
==226054==
==226054== Invalid read of size 8
==226054== at 0x9EFB11: zend_observer_fcall_begin_prechecked (zend_observer.c:269)
==226054== by 0x8E3E8B: zend_observer_fcall_begin_specialized (zend_observer.h:116)
==226054== by 0x8F7BC0: ZEND_DO_FCALL_SPEC_OBSERVER_HANDLER (zend_vm_execute.h:2141)
==226054== by 0x976552: execute_ex (zend_vm_execute.h:58859)
==226054== by 0x97AED8: zend_execute (zend_vm_execute.h:64236)
==226054== by 0xA1B97E: zend_execute_script (zend.c:1934)
==226054== by 0x7CA094: php_execute_script_ex (main.c:2575)
==226054== by 0x7CA212: php_execute_script (main.c:2615)
==226054== by 0xA1DE38: do_cli (php_cli.c:935)
==226054== by 0xA1EB86: main (php_cli.c:1310)
==226054== Address 0x7b0f5d8 is 0 bytes after a block of size 8 alloc'd
==226054== at 0x48457A8: malloc (vg_replace_malloc.c:446)
==226054== by 0x8779E9: __zend_malloc (zend_alloc.c:3280)
==226054== by 0x787C911: preload_load (ZendAccelerator.c:4388)
==226054== by 0x787D3E3: accel_preload (ZendAccelerator.c:4596)
==226054== by 0x787D94F: accel_finish_startup_preload (ZendAccelerator.c:4717)
==226054== by 0x787DDFC: accel_finish_startup (ZendAccelerator.c:4862)
==226054== by 0x7878771: accel_post_startup (ZendAccelerator.c:3373)
==226054== by 0xA19647: zend_post_startup (zend.c:1105)
==226054== by 0x7C99DE: php_module_startup (main.c:2324)
==226054== by 0xA1CD4E: php_cli_startup (php_cli.c:397)
==226054== by 0xA1EAEA: main (php_cli.c:1277)
==226054==
==226054== Invalid read of size 8
==226054== at 0x9EFC94: call_end_observers (zend_observer.c:303)
==226054== by 0x9EFD1C: zend_observer_fcall_end_prechecked (zend_observer.c:315)
==226054== by 0x8E3ED2: zend_observer_fcall_end (zend_observer.h:126)
==226054== by 0x8F7D44: ZEND_DO_FCALL_SPEC_OBSERVER_HANDLER (zend_vm_execute.h:2161)
==226054== by 0x976552: execute_ex (zend_vm_execute.h:58859)
==226054== by 0x97AED8: zend_execute (zend_vm_execute.h:64236)
==226054== by 0xA1B97E: zend_execute_script (zend.c:1934)
==226054== by 0x7CA094: php_execute_script_ex (main.c:2575)
==226054== by 0x7CA212: php_execute_script (main.c:2615)
==226054== by 0xA1DE38: do_cli (php_cli.c:935)
==226054== by 0xA1EB86: main (php_cli.c:1310)
==226054== Address 0x7b0f5d8 is 0 bytes after a block of size 8 alloc'd
==226054== at 0x48457A8: malloc (vg_replace_malloc.c:446)
==226054== by 0x8779E9: __zend_malloc (zend_alloc.c:3280)
==226054== by 0x787C911: preload_load (ZendAccelerator.c:4388)
==226054== by 0x787D3E3: accel_preload (ZendAccelerator.c:4596)
==226054== by 0x787D94F: accel_finish_startup_preload (ZendAccelerator.c:4717)
==226054== by 0x787DDFC: accel_finish_startup (ZendAccelerator.c:4862)
==226054== by 0x7878771: accel_post_startup (ZendAccelerator.c:3373)
==226054== by 0xA19647: zend_post_startup (zend.c:1105)
==226054== by 0x7C99DE: php_module_startup (main.c:2324)
==226054== by 0xA1CD4E: php_cli_startup (php_cli.c:397)
==226054== by 0xA1EAEA: main (php_cli.c:1277)
==226054==
==226054== Invalid read of size 8
==226054== at 0x9EFCA0: call_end_observers (zend_observer.c:303)
==226054== by 0x9EFD1C: zend_observer_fcall_end_prechecked (zend_observer.c:315)
==226054== by 0x8E3ED2: zend_observer_fcall_end (zend_observer.h:126)
==226054== by 0x8F7D44: ZEND_DO_FCALL_SPEC_OBSERVER_HANDLER (zend_vm_execute.h:2161)
==226054== by 0x976552: execute_ex (zend_vm_execute.h:58859)
==226054== by 0x97AED8: zend_execute (zend_vm_execute.h:64236)
==226054== by 0xA1B97E: zend_execute_script (zend.c:1934)
==226054== by 0x7CA094: php_execute_script_ex (main.c:2575)
==226054== by 0x7CA212: php_execute_script (main.c:2615)
==226054== by 0xA1DE38: do_cli (php_cli.c:935)
==226054== by 0xA1EB86: main (php_cli.c:1310)
==226054== Address 0x7b0f5d8 is 0 bytes after a block of size 8 alloc'd
==226054== at 0x48457A8: malloc (vg_replace_malloc.c:446)
==226054== by 0x8779E9: __zend_malloc (zend_alloc.c:3280)
==226054== by 0x787C911: preload_load (ZendAccelerator.c:4388)
==226054== by 0x787D3E3: accel_preload (ZendAccelerator.c:4596)
==226054== by 0x787D94F: accel_finish_startup_preload (ZendAccelerator.c:4717)
==226054== by 0x787DDFC: accel_finish_startup (ZendAccelerator.c:4862)
==226054== by 0x7878771: accel_post_startup (ZendAccelerator.c:3373)
==226054== by 0xA19647: zend_post_startup (zend.c:1105)
==226054== by 0x7C99DE: php_module_startup (main.c:2324)
==226054== by 0xA1CD4E: php_cli_startup (php_cli.c:397)
==226054== by 0xA1EAEA: main (php_cli.c:1277)
==226054==
==226054== Invalid read of size 8
==226054== at 0x9EFCC7: call_end_observers (zend_observer.c:309)
==226054== by 0x9EFD1C: zend_observer_fcall_end_prechecked (zend_observer.c:315)
==226054== by 0x8E3ED2: zend_observer_fcall_end (zend_observer.h:126)
==226054== by 0x8F7D44: ZEND_DO_FCALL_SPEC_OBSERVER_HANDLER (zend_vm_execute.h:2161)
==226054== by 0x976552: execute_ex (zend_vm_execute.h:58859)
==226054== by 0x97AED8: zend_execute (zend_vm_execute.h:64236)
==226054== by 0xA1B97E: zend_execute_script (zend.c:1934)
==226054== by 0x7CA094: php_execute_script_ex (main.c:2575)
==226054== by 0x7CA212: php_execute_script (main.c:2615)
==226054== by 0xA1DE38: do_cli (php_cli.c:935)
==226054== by 0xA1EB86: main (php_cli.c:1310)
==226054== Address 0x7b0f5d8 is 0 bytes after a block of size 8 alloc'd
==226054== at 0x48457A8: malloc (vg_replace_malloc.c:446)
==226054== by 0x8779E9: __zend_malloc (zend_alloc.c:3280)
==226054== by 0x787C911: preload_load (ZendAccelerator.c:4388)
==226054== by 0x787D3E3: accel_preload (ZendAccelerator.c:4596)
==226054== by 0x787D94F: accel_finish_startup_preload (ZendAccelerator.c:4717)
==226054== by 0x787DDFC: accel_finish_startup (ZendAccelerator.c:4862)
==226054== by 0x7878771: accel_post_startup (ZendAccelerator.c:3373)
==226054== by 0xA19647: zend_post_startup (zend.c:1105)
==226054== by 0x7C99DE: php_module_startup (main.c:2324)
==226054== by 0xA1CD4E: php_cli_startup (php_cli.c:397)
==226054== by 0xA1EAEA: main (php_cli.c:1277)
This was quite worth the cost, yes :-(
I'm not happy about the extra complexity and may have to think about streamlining this better. But that's not for now.
Is it still worth the cost? I'm not convinced tbh. I rather revert the original PR. The public ZEND_API fields introduced by the original PR can be hardcoded to zeros.
Signed-off-by: Bob Weinand <[email protected]>
@nielsdos You're absolutely right. It passed under asan, because asan is 16 byte aligned. Accidentally multiplied by If I recall correctly this change had a minor (0.1~0.2%) benefit to performance without observers and 1.5% benefit with observers enabled with symfony demo. I'd be honestly disappointed if rolling it back were the way to go. |
Signed-off-by: Bob Weinand <[email protected]>
This solely affects the builtin enum functions currently.
Given that these are stored in SHM, we cannot simply hardwire a pointer into the internal function runtime cache on NTS too, but have to use a MAP_PTR (like on ZTS). Now, by design, the runtime cache of internal functions no longer is reset between requests, hence we need to store them explicitly as static runtime cache.
On NTS builds we cannot trivially move the pointers into CG(internal_run_time_cache) as they're directly stored on the individual functions (on ZTS we could simply iterate the static map_ptrs). Hence, we have the choice between having opcache managing the internal run_time_cache for its preloaded functions itself or realloc CG(internal_run_time_cache) and iterate through all functions to assign the new address. We choose the latter for simplicity and initial speed.
This changes the accel globals structs, but that should be fine given that they're not exported.
map_ptr_static_last has been added as last element to zend_accel_shared_globals, so that accesses to it are compatible. We do not have to care about the ABI of creating new zend_accel_shared_globals structs. That's opcaches prerogative.