php-internal-docs 8.4.8
Unofficial docs for php/php-src
Loading...
Searching...
No Matches
fiber.c
Go to the documentation of this file.
1/*
2 +----------------------------------------------------------------------+
3 | Copyright (c) The PHP Group |
4 +----------------------------------------------------------------------+
5 | This source file is subject to version 3.01 of the PHP license, |
6 | that is bundled with this package in the file LICENSE, and is |
7 | available through the world-wide-web at the following url: |
8 | https://www.php.net/license/3_01.txt |
9 | If you did not receive a copy of the PHP license and are unable to |
10 | obtain it through the world-wide-web, please send a note to |
11 | license@php.net so we can mail you a copy immediately. |
12 +----------------------------------------------------------------------+
13 | Authors: Aaron Piotrowski <aaron@trowski.com> |
14 +----------------------------------------------------------------------+
15*/
16
17#include "php_test.h"
18#include "fiber.h"
19#include "fiber_arginfo.h"
20#include "zend_fibers.h"
21#include "zend_exceptions.h"
22
23static zend_class_entry *zend_test_fiber_class;
24static zend_object_handlers zend_test_fiber_handlers;
25
26static zend_fiber_transfer zend_test_fiber_switch_to(zend_fiber_context *context, zval *value, bool exception)
27{
28 zend_fiber_transfer transfer = {
29 .context = context,
31 };
32
33 if (value) {
34 ZVAL_COPY(&transfer.value, value);
35 } else {
36 ZVAL_NULL(&transfer.value);
37 }
38
40
41 /* Forward bailout into current fiber. */
44 }
45
46 return transfer;
47}
48
49static zend_fiber_transfer zend_test_fiber_resume(zend_test_fiber *fiber, zval *value, bool exception)
50{
52
53 fiber->caller = EG(current_fiber_context);
54 ZT_G(active_fiber) = fiber;
55
56 zend_fiber_transfer transfer = zend_test_fiber_switch_to(fiber->previous, value, exception);
57
58 ZT_G(active_fiber) = previous;
59
60 return transfer;
61}
62
63static zend_fiber_transfer zend_test_fiber_suspend(zend_test_fiber *fiber, zval *value)
64{
65 ZEND_ASSERT(fiber->caller != NULL);
66
67 zend_fiber_context *caller = fiber->caller;
68 fiber->previous = EG(current_fiber_context);
69 fiber->caller = NULL;
70
71 return zend_test_fiber_switch_to(caller, value, false);
72}
73
74static ZEND_STACK_ALIGNED void zend_test_fiber_execute(zend_fiber_transfer *transfer)
75{
78
80
81 EG(vm_stack) = NULL;
82 transfer->flags = 0;
83
85 zend_vm_stack stack = zend_vm_stack_new_page(ZEND_FIBER_VM_STACK_SIZE, NULL);
86 EG(vm_stack) = stack;
87 EG(vm_stack_top) = stack->top + ZEND_CALL_FRAME_SLOT;
88 EG(vm_stack_end) = stack->end;
89 EG(vm_stack_page_size) = ZEND_FIBER_VM_STACK_SIZE;
90
92
95
96 EG(current_execute_data) = execute_data;
97 EG(jit_trace_num) = 0;
98
99#ifdef ZEND_CHECK_STACK_LIMIT
100 EG(stack_base) = zend_fiber_stack_base(fiber->context.stack);
101 EG(stack_limit) = zend_fiber_stack_limit(fiber->context.stack);
102#endif
103 fiber->fci.retval = &retval;
104
105 zend_call_function(&fiber->fci, &fiber->fci_cache);
106
107 zval_ptr_dtor(&fiber->result); // Destroy param from symmetric coroutine.
109
110 if (EG(exception)) {
111 if (!(fiber->flags & ZEND_FIBER_FLAG_DESTROYED)
113 ) {
116
117 ZVAL_OBJ_COPY(&transfer->value, EG(exception));
118 }
119
121 } else {
122 ZVAL_COPY_VALUE(&fiber->result, &retval);
123 ZVAL_COPY(&transfer->value, &fiber->result);
124 }
125 } zend_catch {
128 } zend_end_try();
129
131
132 if (fiber->target) {
133 zend_fiber_context *target = &fiber->target->context;
134 zend_fiber_init_context(target, zend_test_fiber_class, zend_test_fiber_execute, EG(fiber_stack_size));
135 transfer->context = target;
136
137 ZVAL_COPY(&fiber->target->result, &fiber->result);
138 fiber->target->fci.params = &fiber->target->result;
139 fiber->target->fci.param_count = 1;
140
141 fiber->target->caller = fiber->caller;
142 ZT_G(active_fiber) = fiber->target;
143 } else {
144 transfer->context = fiber->caller;
145 }
146
147 fiber->caller = NULL;
148}
149
150static zend_object *zend_test_fiber_object_create(zend_class_entry *ce)
151{
152 zend_test_fiber *fiber;
153
154 fiber = emalloc(sizeof(zend_test_fiber));
155 memset(fiber, 0, sizeof(zend_test_fiber));
156
157 zend_object_std_init(&fiber->std, ce);
158
159 return &fiber->std;
160}
161
162static void zend_test_fiber_object_destroy(zend_object *object)
163{
164 zend_test_fiber *fiber = (zend_test_fiber *) object;
165
167 return;
168 }
169
171 EG(exception) = NULL;
172
174
175 zend_fiber_transfer transfer = zend_test_fiber_resume(fiber, NULL, false);
176
177 if (transfer.flags & ZEND_FIBER_TRANSFER_FLAG_ERROR) {
178 EG(exception) = Z_OBJ(transfer.value);
179
180 if (!exception && EG(current_execute_data) && EG(current_execute_data)->func
181 && ZEND_USER_CODE(EG(current_execute_data)->func->common.type)) {
182 zend_rethrow_exception(EG(current_execute_data));
183 }
184
186
187 if (!EG(current_execute_data)) {
189 }
190 } else {
191 zval_ptr_dtor(&transfer.value);
193 }
194}
195
196static void zend_test_fiber_object_free(zend_object *object)
197{
198 zend_test_fiber *fiber = (zend_test_fiber *) object;
199
200 if (fiber->context.status == ZEND_FIBER_STATUS_INIT) {
201 // Fiber was never started, so we need to release the reference to the callback.
203 }
204
205 if (fiber->target) {
206 OBJ_RELEASE(&fiber->target->std);
207 }
208
209 zval_ptr_dtor(&fiber->result);
210
211 zend_object_std_dtor(&fiber->std);
212}
213
214static zend_always_inline void delegate_transfer_result(
216) {
217 if (transfer->flags & ZEND_FIBER_TRANSFER_FLAG_ERROR) {
220 }
221
222 if (fiber->context.status == ZEND_FIBER_STATUS_DEAD) {
223 zval_ptr_dtor(&transfer->value);
224 RETURN_NULL();
225 }
226
227 RETURN_COPY_VALUE(&transfer->value);
228}
229
230static ZEND_METHOD(_ZendTestFiber, __construct)
231{
233
235 Z_PARAM_FUNC(fiber->fci, fiber->fci_cache)
237
238 // Keep a reference to closures or callable objects while the fiber is running.
239 Z_TRY_ADDREF(fiber->fci.function_name);
240}
241
243{
245 zval *params;
246 uint32_t param_count;
247 zend_array *named_params;
248
250 Z_PARAM_VARIADIC_WITH_NAMED(params, param_count, named_params);
252
254
255 if (fiber->previous != NULL) {
256 zend_throw_error(NULL, "Cannot start a fiber that is the target of another fiber");
258 }
259
260 fiber->fci.params = params;
261 fiber->fci.param_count = param_count;
262 fiber->fci.named_params = named_params;
263
264 zend_fiber_init_context(&fiber->context, zend_test_fiber_class, zend_test_fiber_execute, EG(fiber_stack_size));
265
266 fiber->previous = &fiber->context;
267
268 zend_fiber_transfer transfer = zend_test_fiber_resume(fiber, NULL, false);
269
270 delegate_transfer_result(fiber, &transfer, INTERNAL_FUNCTION_PARAM_PASSTHRU);
271}
272
273static ZEND_METHOD(_ZendTestFiber, suspend)
274{
275 zval *value = NULL;
276
281
283
284 ZEND_ASSERT(fiber);
285
286 zend_fiber_transfer transfer = zend_test_fiber_suspend(fiber, value);
287
289 // This occurs when the test fiber is GC'ed while suspended.
290 zval_ptr_dtor(&transfer.value);
293 }
294
295 delegate_transfer_result(fiber, &transfer, INTERNAL_FUNCTION_PARAM_PASSTHRU);
296}
297
298static ZEND_METHOD(_ZendTestFiber, resume)
299{
300 zend_test_fiber *fiber;
301 zval *value = NULL;
302
307
308 fiber = (zend_test_fiber *) Z_OBJ_P(ZEND_THIS);
309
310 if (UNEXPECTED(fiber->context.status != ZEND_FIBER_STATUS_SUSPENDED || fiber->caller != NULL)) {
311 zend_throw_error(NULL, "Cannot resume a fiber that is not suspended");
313 }
314
315 zend_fiber_transfer transfer = zend_test_fiber_resume(fiber, value, false);
316
317 delegate_transfer_result(fiber, &transfer, INTERNAL_FUNCTION_PARAM_PASSTHRU);
318}
319
320static ZEND_METHOD(_ZendTestFiber, pipeTo)
321{
322 zend_fcall_info fci;
323 zend_fcall_info_cache fci_cache;
324
326 Z_PARAM_FUNC(fci, fci_cache)
328
330 zend_test_fiber *target = (zend_test_fiber *) zend_test_fiber_class->create_object(zend_test_fiber_class);
331
332 target->fci = fci;
333 target->fci_cache = fci_cache;
334 Z_TRY_ADDREF(target->fci.function_name);
335
336 target->previous = &fiber->context;
337
338 if (fiber->target) {
339 OBJ_RELEASE(&fiber->target->std);
340 }
341
342 fiber->target = target;
343
344 RETURN_OBJ_COPY(&target->std);
345}
346
348{
349 zend_test_fiber_class = register_class__ZendTestFiber();
350 zend_test_fiber_class->create_object = zend_test_fiber_object_create;
351 zend_test_fiber_class->default_object_handlers = &zend_test_fiber_handlers;
352
353 zend_test_fiber_handlers = std_object_handlers;
354 zend_test_fiber_handlers.dtor_obj = zend_test_fiber_object_destroy;
355 zend_test_fiber_handlers.free_obj = zend_test_fiber_object_free;
356 zend_test_fiber_handlers.clone_obj = NULL;
357}
bool exception
Definition assert.c:30
DNS_STATUS status
Definition dns_win32.c:49
memset(ptr, 0, type->size)
buf start
Definition ffi.c:4687
void zend_test_fiber_init(void)
Definition fiber.c:347
struct _zend_test_fiber zend_test_fiber
Definition fiber.h:22
#define NULL
Definition gdcache.h:45
foreach($dp as $el) foreach( $dp as $el) if( $pass2< 2) echo ""
PdoFirebird_ce create_object
#define ZT_G(v)
zend_test_fiber * active_fiber
Definition php_test.h:61
HashTable * named_params
Definition zend_API.h:56
uint32_t param_count
Definition zend_API.h:51
zend_fiber_status status
Definition zend_fibers.h:93
zend_fiber_stack * stack
Definition zend_fibers.h:90
zend_fiber_context * context
Definition zend_fibers.h:62
zend_fiber_context * caller
Definition fiber.h:28
zend_fiber_context * previous
Definition fiber.h:29
zend_object std
Definition fiber.h:25
uint8_t flags
Definition fiber.h:26
zend_fiber_context context
Definition fiber.h:27
zend_fcall_info fci
Definition fiber.h:31
zend_test_fiber * target
Definition fiber.h:30
zend_fcall_info_cache fci_cache
Definition fiber.h:32
Definition dce.c:49
ZEND_API ZEND_COLD void zend_throw_error(zend_class_entry *exception_ce, const char *format,...)
Definition zend.c:1772
#define INTERNAL_FUNCTION_PARAMETERS
Definition zend.h:49
#define zend_first_try
Definition zend.h:284
#define zend_catch
Definition zend.h:277
#define zend_end_try()
Definition zend.h:280
#define zend_bailout()
Definition zend.h:268
#define INTERNAL_FUNCTION_PARAM_PASSTHRU
Definition zend.h:50
#define Z_PARAM_FUNC(dest_fci, dest_fcc)
Definition zend_API.h:1824
struct _zend_fcall_info_cache zend_fcall_info_cache
#define ZEND_PARSE_PARAMETERS_END()
Definition zend_API.h:1641
#define RETURN_NULL()
Definition zend_API.h:1036
#define Z_PARAM_OPTIONAL
Definition zend_API.h:1667
#define ZEND_PARSE_PARAMETERS_START(min_num_args, max_num_args)
Definition zend_API.h:1620
#define RETURN_OBJ_COPY(r)
Definition zend_API.h:1053
struct _zend_fcall_info zend_fcall_info
#define ZEND_METHOD(classname, name)
Definition zend_API.h:76
#define RETURN_THROWS()
Definition zend_API.h:1060
#define ZEND_THIS
Definition zend_API.h:523
#define RETURN_COPY_VALUE(zv)
Definition zend_API.h:1055
#define Z_PARAM_ZVAL(dest)
Definition zend_API.h:2100
ZEND_API zend_result zend_call_function(zend_fcall_info *fci, zend_fcall_info_cache *fci_cache)
#define Z_PARAM_VARIADIC_WITH_NAMED(dest, dest_num, dest_named)
Definition zend_API.h:2127
#define emalloc(size)
Definition zend_alloc.h:151
struct _zval_struct zval
execute_data func
#define ZEND_USER_CODE(type)
#define ZEND_CALL_FRAME_SLOT
#define E_ERROR
Definition zend_errors.h:23
ZEND_API ZEND_COLD void zend_throw_exception_internal(zend_object *exception)
ZEND_API bool zend_is_unwind_exit(const zend_object *ex)
ZEND_API bool zend_is_graceful_exit(const zend_object *ex)
ZEND_API ZEND_COLD zend_result zend_exception_error(zend_object *ex, int severity)
ZEND_API ZEND_COLD void zend_throw_graceful_exit(void)
ZEND_API void zend_clear_exception(void)
void zend_exception_set_previous(zend_object *exception, zend_object *add_previous)
ZEND_API const zend_internal_function zend_pass_function
ZEND_API void zend_vm_stack_destroy(void)
ZEND_API zend_result zend_fiber_init_context(zend_fiber_context *context, void *kind, zend_fiber_coroutine coroutine, size_t stack_size)
ZEND_API void zend_fiber_switch_context(zend_fiber_transfer *transfer)
@ ZEND_FIBER_TRANSFER_FLAG_BAILOUT
Definition zend_fibers.h:48
@ ZEND_FIBER_TRANSFER_FLAG_ERROR
Definition zend_fibers.h:47
@ ZEND_FIBER_FLAG_THREW
Definition zend_fibers.h:41
@ ZEND_FIBER_FLAG_DESTROYED
Definition zend_fibers.h:43
@ ZEND_FIBER_FLAG_BAILOUT
Definition zend_fibers.h:42
@ ZEND_FIBER_STATUS_INIT
Definition zend_fibers.h:34
@ ZEND_FIBER_STATUS_SUSPENDED
Definition zend_fibers.h:36
@ ZEND_FIBER_STATUS_DEAD
Definition zend_fibers.h:37
struct _zend_fiber_transfer zend_fiber_transfer
#define ZEND_FIBER_VM_STACK_SIZE
Definition zend_fibers.h:29
union _zend_function zend_function
struct _zend_fiber_context zend_fiber_context
struct _zend_vm_stack * zend_vm_stack
#define EG(v)
ZEND_API const zend_object_handlers std_object_handlers
ZEND_API void ZEND_FASTCALL zend_object_std_init(zend_object *object, zend_class_entry *ce)
ZEND_API void zend_object_std_dtor(zend_object *object)
#define OBJ_RELEASE(obj)
#define zend_always_inline
#define ZEND_STACK_ALIGNED
#define ZEND_ASSERT(c)
#define UNEXPECTED(condition)
struct _zend_array zend_array
struct _zend_class_entry zend_class_entry
struct _zend_object zend_object
#define ZVAL_NULL(z)
#define Z_OBJ_P(zval_p)
Definition zend_types.h:990
#define Z_TRY_ADDREF(z)
#define ZVAL_COPY(z, v)
#define ZVAL_OBJ_COPY(z, o)
struct _zend_object_handlers zend_object_handlers
Definition zend_types.h:88
struct _zend_execute_data zend_execute_data
Definition zend_types.h:91
#define ZVAL_COPY_VALUE(z, v)
#define Z_OBJ(zval)
Definition zend_types.h:989
ZEND_API void zval_ptr_dtor(zval *zval_ptr)
zval retval
execute_data
value