rfc:zend-vm-pause-api
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
rfc:zend-vm-pause-api [2017/11/01 01:25] – lvht | rfc:zend-vm-pause-api [2025/04/03 13:08] (current) – external edit 127.0.0.1 | ||
---|---|---|---|
Line 2: | Line 2: | ||
* Version: 0.9 | * Version: 0.9 | ||
* Date: 2017-11-01 | * Date: 2017-11-01 | ||
- | * Author: Haitao Lv php@lvht.net | + | * Author: Haitao Lv, php@lvht.net |
- | * Status: | + | * Status: |
* First Published at: http:// | * First Published at: http:// | ||
+ | |||
+ | This RPC has been obsolete. Please see https:// | ||
===== Introduction ===== | ===== Introduction ===== | ||
Line 14: | Line 16: | ||
===== Proposal ===== | ===== Proposal ===== | ||
- | This PRFC propose a new zend_vm_pause API and an new EG(pause_op), by which some zend extension could make the | + | This PRFC propose a new vm_interrupt type, by which some zend extension could make the zend vm execution |
- | zend vm execution | + | So we can implement feature like Fiber in a standalone extension. |
- | We need introduce a new vm global EG(pause_op) as zend_op and init it by the following code | ||
<code c> | <code c> | ||
- | static void zend_init_pause_op(void) | + | diff --git a/ |
- | { | + | index 183072033607..bd7408e824fc 100644 |
- | memset(& | + | --- a/Zend/zend_vm_def.h |
- | + | +++ b/Zend/zend_vm_def.h | |
- | EG(pause_op).opcode = ZEND_HANDLE_PAUSE; | + | @@ -8893,13 +8893,19 @@ ZEND_VM_DEFINE_OP(137, ZEND_OP_DATA); |
- | EG(pause_op).op1_type = IS_UNUSED; | + | |
- | EG(pause_op).op2_type = IS_UNUSED; | + | ZEND_VM_HELPER(zend_interrupt_helper, ANY, ANY) |
- | | + | { |
- | + | + int8_t interrupt_type | |
- | ZEND_VM_SET_OPCODE_HANDLER(& | + | + |
- | } | + | EG(vm_interrupt) = 0; |
- | </ | + | if (EG(timed_out)) { |
- | + | zend_timeout(0); | |
- | A new opcode ZEND_HANDLE_PAUSE will also be need, and its handle looks like | + | } else if (zend_interrupt_function) { |
- | <code c> | + | SAVE_OPLINE(); |
- | ZEND_VM_HANDLER(198, | + | zend_interrupt_function(execute_data); |
- | { | + | - ZEND_VM_ENTER(); |
- | void (*fn)(const zend_op *opline, zend_execute_data *execute_data); | + | + if (UNEXPECTED(interrupt_type |
- | + | + ZEND_VM_RETURN(); | |
- | USE_OPLINE; | + | + } else { |
- | + | + ZEND_VM_ENTER(); | |
- | // use pause_op' | + | + } |
- | // so we can do something in extension before paused | + | } |
- | fn = *((void**)& | + | ZEND_VM_CONTINUE(); |
- | + | } | |
- | // restore the opline pointer | + | |
- | opline = *((const zend_op**)& | + | |
- | + | ||
- | // execute the callback function | + | |
- | if (EXPECTED(fn != NULL)) { | + | |
- | fn(opline, execute_data); | + | |
- | } | + | |
- | + | ||
- | SAVE_OPLINE(); | + | |
- | + | ||
- | // make vm paused here | + | |
- | ZEND_VM_RETURN(); | + | |
- | } | + | |
- | </ | + | |
- | + | ||
- | The zend_vm_pause API is like this | + | |
- | <code c> | + | |
- | ZEND_API ZEND_COLD void zend_vm_pause(void (*fn)(const zend_op*, zend_execute_data*)) /* {{{ */ | + | |
- | { | + | |
- | zend_execute_data *execute_data | + | |
- | + | ||
- | // use EG(paused_op).op1 and EG(paused_op).op2 to save the next opline pointer | + | |
- | // and the ZEND_HANDLE_PAUSE will resume opline to this | + | |
- | *((const zend_op**)& | + | |
- | + | ||
- | // use EG(paused_op).result and EG(paused_op).extended_value to save the callback function pointer | + | |
- | *((void**)& | + | |
- | + | ||
- | // make the next opline point to EG(pause_op) | + | |
- | // so that Zend VM will execute it after the DO_FCALL opline | + | |
- | EX(prev_execute_data)-> | + | |
- | } | + | |
- | </ | + | |
- | + | ||
- | You will see that we want to make the Zend VM execute the EG(pause_op) by changing the EX(prev_execute_data)-> | + | |
- | + | ||
- | However, it does not work out of box. The reason is under the ZEND_DO_FCALL' | + | |
- | <code c> | + | |
- | ZEND_VM_HOT_HANDLER(60, ZEND_DO_FCALL, | + | |
- | { | + | |
- | // ... | + | |
- | + | ||
- | // load opline from EG(current_execute_data).opline | + | |
- | | + | |
- | + | ||
- | | + | |
- | // ... | + | |
- | } else if (EXPECTED(fbc-> | + | |
- | // ... | + | |
- | + | ||
- | // the extension' | + | |
- | // after its execution, we changed the EG(current_execute_data).opline | + | |
- | // However, as the value of EG(current_execute_data).opline has already been loade into the opline | + | |
- | + | ||
- | // We need to reload the opline | + | |
- | LOAD_OPLINE(); // load the opline again | + | |
- | + | ||
- | // ... | + | |
- | | + | |
- | // ... | + | |
- | } | + | |
- | + | ||
- | // ... | + | |
- | + | ||
- | // so the opline | + | |
- | | + | |
- | ZEND_VM_CONTINUE(); | + | |
- | } | + | |
</ | </ | ||
Line 123: | Line 56: | ||
===== RFC Impact ===== | ===== RFC Impact ===== | ||
- | Any internal function call will LOAD_OPLINE() twice. And this will make slow the execution, theoretically. | + | None |
==== To SAPIs ==== | ==== To SAPIs ==== | ||
None | None | ||
Line 154: | Line 87: | ||
===== Patches and Tests ===== | ===== Patches and Tests ===== | ||
- | https:// | + | https:// |
===== Implementation ===== | ===== Implementation ===== |
rfc/zend-vm-pause-api.1509499514.txt.gz · Last modified: 2025/04/03 13:08 (external edit)