fiber.c 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  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. #include "php_test.h"
  17. #include "fiber.h"
  18. #include "fiber_arginfo.h"
  19. #include "zend_fibers.h"
  20. #include "zend_exceptions.h"
  21. static zend_class_entry *zend_test_fiber_class;
  22. static zend_object_handlers zend_test_fiber_handlers;
  23. static zend_fiber_transfer zend_test_fiber_switch_to(zend_fiber_context *context, zval *value, bool exception)
  24. {
  25. zend_fiber_transfer transfer = {
  26. .context = context,
  27. .flags = exception ? ZEND_FIBER_TRANSFER_FLAG_ERROR : 0,
  28. };
  29. if (value) {
  30. ZVAL_COPY(&transfer.value, value);
  31. } else {
  32. ZVAL_NULL(&transfer.value);
  33. }
  34. zend_fiber_switch_context(&transfer);
  35. /* Forward bailout into current fiber. */
  36. if (UNEXPECTED(transfer.flags & ZEND_FIBER_TRANSFER_FLAG_BAILOUT)) {
  37. zend_bailout();
  38. }
  39. return transfer;
  40. }
  41. static zend_fiber_transfer zend_test_fiber_resume(zend_test_fiber *fiber, zval *value, bool exception)
  42. {
  43. zend_test_fiber *previous = ZT_G(active_fiber);
  44. fiber->caller = EG(current_fiber_context);
  45. ZT_G(active_fiber) = fiber;
  46. zend_fiber_transfer transfer = zend_test_fiber_switch_to(fiber->previous, value, exception);
  47. ZT_G(active_fiber) = previous;
  48. return transfer;
  49. }
  50. static zend_fiber_transfer zend_test_fiber_suspend(zend_test_fiber *fiber, zval *value)
  51. {
  52. ZEND_ASSERT(fiber->caller != NULL);
  53. zend_fiber_context *caller = fiber->caller;
  54. fiber->previous = EG(current_fiber_context);
  55. fiber->caller = NULL;
  56. return zend_test_fiber_switch_to(caller, value, false);
  57. }
  58. static ZEND_STACK_ALIGNED void zend_test_fiber_execute(zend_fiber_transfer *transfer)
  59. {
  60. zend_test_fiber *fiber = ZT_G(active_fiber);
  61. zval retval;
  62. zend_execute_data *execute_data;
  63. EG(vm_stack) = NULL;
  64. transfer->flags = 0;
  65. zend_first_try {
  66. zend_vm_stack stack = zend_vm_stack_new_page(ZEND_FIBER_VM_STACK_SIZE, NULL);
  67. EG(vm_stack) = stack;
  68. EG(vm_stack_top) = stack->top + ZEND_CALL_FRAME_SLOT;
  69. EG(vm_stack_end) = stack->end;
  70. EG(vm_stack_page_size) = ZEND_FIBER_VM_STACK_SIZE;
  71. execute_data = (zend_execute_data *) stack->top;
  72. memset(execute_data, 0, sizeof(zend_execute_data));
  73. EG(current_execute_data) = execute_data;
  74. EG(jit_trace_num) = 0;
  75. fiber->fci.retval = &retval;
  76. zend_call_function(&fiber->fci, &fiber->fci_cache);
  77. zval_ptr_dtor(&fiber->result); // Destroy param from symmetric coroutine.
  78. zval_ptr_dtor(&fiber->fci.function_name);
  79. if (EG(exception)) {
  80. if (!(fiber->flags & ZEND_FIBER_FLAG_DESTROYED)
  81. || !(zend_is_graceful_exit(EG(exception)) || zend_is_unwind_exit(EG(exception)))
  82. ) {
  83. fiber->flags |= ZEND_FIBER_FLAG_THREW;
  84. transfer->flags = ZEND_FIBER_TRANSFER_FLAG_ERROR;
  85. ZVAL_OBJ_COPY(&transfer->value, EG(exception));
  86. }
  87. zend_clear_exception();
  88. } else {
  89. ZVAL_COPY_VALUE(&fiber->result, &retval);
  90. ZVAL_COPY(&transfer->value, &fiber->result);
  91. }
  92. } zend_catch {
  93. fiber->flags |= ZEND_FIBER_FLAG_BAILOUT;
  94. transfer->flags = ZEND_FIBER_TRANSFER_FLAG_BAILOUT;
  95. } zend_end_try();
  96. zend_vm_stack_destroy();
  97. if (fiber->target) {
  98. zend_fiber_context *target = &fiber->target->context;
  99. zend_fiber_init_context(target, zend_test_fiber_class, zend_test_fiber_execute, EG(fiber_stack_size));
  100. transfer->context = target;
  101. ZVAL_COPY(&fiber->target->result, &fiber->result);
  102. fiber->target->fci.params = &fiber->target->result;
  103. fiber->target->fci.param_count = 1;
  104. fiber->target->caller = fiber->caller;
  105. ZT_G(active_fiber) = fiber->target;
  106. } else {
  107. transfer->context = fiber->caller;
  108. }
  109. fiber->caller = NULL;
  110. }
  111. static zend_object *zend_test_fiber_object_create(zend_class_entry *ce)
  112. {
  113. zend_test_fiber *fiber;
  114. fiber = emalloc(sizeof(zend_test_fiber));
  115. memset(fiber, 0, sizeof(zend_test_fiber));
  116. zend_object_std_init(&fiber->std, ce);
  117. fiber->std.handlers = &zend_test_fiber_handlers;
  118. return &fiber->std;
  119. }
  120. static void zend_test_fiber_object_destroy(zend_object *object)
  121. {
  122. zend_test_fiber *fiber = (zend_test_fiber *) object;
  123. if (fiber->context.status != ZEND_FIBER_STATUS_SUSPENDED) {
  124. return;
  125. }
  126. zend_object *exception = EG(exception);
  127. EG(exception) = NULL;
  128. fiber->flags |= ZEND_FIBER_FLAG_DESTROYED;
  129. zend_fiber_transfer transfer = zend_test_fiber_resume(fiber, NULL, false);
  130. if (transfer.flags & ZEND_FIBER_TRANSFER_FLAG_ERROR) {
  131. EG(exception) = Z_OBJ(transfer.value);
  132. if (!exception && EG(current_execute_data) && EG(current_execute_data)->func
  133. && ZEND_USER_CODE(EG(current_execute_data)->func->common.type)) {
  134. zend_rethrow_exception(EG(current_execute_data));
  135. }
  136. zend_exception_set_previous(EG(exception), exception);
  137. if (!EG(current_execute_data)) {
  138. zend_exception_error(EG(exception), E_ERROR);
  139. }
  140. } else {
  141. zval_ptr_dtor(&transfer.value);
  142. EG(exception) = exception;
  143. }
  144. }
  145. static void zend_test_fiber_object_free(zend_object *object)
  146. {
  147. zend_test_fiber *fiber = (zend_test_fiber *) object;
  148. if (fiber->context.status == ZEND_FIBER_STATUS_INIT) {
  149. // Fiber was never started, so we need to release the reference to the callback.
  150. zval_ptr_dtor(&fiber->fci.function_name);
  151. }
  152. if (fiber->target) {
  153. OBJ_RELEASE(&fiber->target->std);
  154. }
  155. zval_ptr_dtor(&fiber->result);
  156. zend_object_std_dtor(&fiber->std);
  157. }
  158. static zend_always_inline void delegate_transfer_result(
  159. zend_test_fiber *fiber, zend_fiber_transfer *transfer, INTERNAL_FUNCTION_PARAMETERS
  160. ) {
  161. if (transfer->flags & ZEND_FIBER_TRANSFER_FLAG_ERROR) {
  162. zend_throw_exception_internal(Z_OBJ(transfer->value));
  163. RETURN_THROWS();
  164. }
  165. if (fiber->context.status == ZEND_FIBER_STATUS_DEAD) {
  166. zval_ptr_dtor(&transfer->value);
  167. RETURN_NULL();
  168. }
  169. RETURN_COPY_VALUE(&transfer->value);
  170. }
  171. static ZEND_METHOD(_ZendTestFiber, __construct)
  172. {
  173. zend_test_fiber *fiber = (zend_test_fiber *) Z_OBJ_P(getThis());
  174. ZEND_PARSE_PARAMETERS_START(1, 1)
  175. Z_PARAM_FUNC(fiber->fci, fiber->fci_cache)
  176. ZEND_PARSE_PARAMETERS_END();
  177. // Keep a reference to closures or callable objects while the fiber is running.
  178. Z_TRY_ADDREF(fiber->fci.function_name);
  179. }
  180. static ZEND_METHOD(_ZendTestFiber, start)
  181. {
  182. zend_test_fiber *fiber = (zend_test_fiber *) Z_OBJ_P(getThis());
  183. zval *params;
  184. uint32_t param_count;
  185. zend_array *named_params;
  186. ZEND_PARSE_PARAMETERS_START(0, -1)
  187. Z_PARAM_VARIADIC_WITH_NAMED(params, param_count, named_params);
  188. ZEND_PARSE_PARAMETERS_END();
  189. ZEND_ASSERT(fiber->context.status == ZEND_FIBER_STATUS_INIT);
  190. if (fiber->previous != NULL) {
  191. zend_throw_error(NULL, "Cannot start a fiber that is the target of another fiber");
  192. RETURN_THROWS();
  193. }
  194. fiber->fci.params = params;
  195. fiber->fci.param_count = param_count;
  196. fiber->fci.named_params = named_params;
  197. zend_fiber_init_context(&fiber->context, zend_test_fiber_class, zend_test_fiber_execute, EG(fiber_stack_size));
  198. fiber->previous = &fiber->context;
  199. zend_fiber_transfer transfer = zend_test_fiber_resume(fiber, NULL, false);
  200. delegate_transfer_result(fiber, &transfer, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  201. }
  202. static ZEND_METHOD(_ZendTestFiber, suspend)
  203. {
  204. zval *value = NULL;
  205. ZEND_PARSE_PARAMETERS_START(0, 1)
  206. Z_PARAM_OPTIONAL
  207. Z_PARAM_ZVAL(value);
  208. ZEND_PARSE_PARAMETERS_END();
  209. zend_test_fiber *fiber = ZT_G(active_fiber);
  210. ZEND_ASSERT(fiber);
  211. zend_fiber_transfer transfer = zend_test_fiber_suspend(fiber, value);
  212. if (fiber->flags & ZEND_FIBER_FLAG_DESTROYED) {
  213. // This occurs when the test fiber is GC'ed while suspended.
  214. zval_ptr_dtor(&transfer.value);
  215. zend_throw_graceful_exit();
  216. RETURN_THROWS();
  217. }
  218. delegate_transfer_result(fiber, &transfer, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  219. }
  220. static ZEND_METHOD(_ZendTestFiber, resume)
  221. {
  222. zend_test_fiber *fiber;
  223. zval *value = NULL;
  224. ZEND_PARSE_PARAMETERS_START(0, 1)
  225. Z_PARAM_OPTIONAL
  226. Z_PARAM_ZVAL(value);
  227. ZEND_PARSE_PARAMETERS_END();
  228. fiber = (zend_test_fiber *) Z_OBJ_P(getThis());
  229. if (UNEXPECTED(fiber->context.status != ZEND_FIBER_STATUS_SUSPENDED || fiber->caller != NULL)) {
  230. zend_throw_error(NULL, "Cannot resume a fiber that is not suspended");
  231. RETURN_THROWS();
  232. }
  233. zend_fiber_transfer transfer = zend_test_fiber_resume(fiber, value, false);
  234. delegate_transfer_result(fiber, &transfer, INTERNAL_FUNCTION_PARAM_PASSTHRU);
  235. }
  236. static ZEND_METHOD(_ZendTestFiber, pipeTo)
  237. {
  238. zend_fcall_info fci;
  239. zend_fcall_info_cache fci_cache;
  240. ZEND_PARSE_PARAMETERS_START(1, 1)
  241. Z_PARAM_FUNC(fci, fci_cache)
  242. ZEND_PARSE_PARAMETERS_END();
  243. zend_test_fiber *fiber = (zend_test_fiber *) Z_OBJ_P(getThis());
  244. zend_test_fiber *target = (zend_test_fiber *) zend_test_fiber_class->create_object(zend_test_fiber_class);
  245. target->fci = fci;
  246. target->fci_cache = fci_cache;
  247. Z_TRY_ADDREF(target->fci.function_name);
  248. target->previous = &fiber->context;
  249. if (fiber->target) {
  250. OBJ_RELEASE(&fiber->target->std);
  251. }
  252. fiber->target = target;
  253. RETURN_OBJ_COPY(&target->std);
  254. }
  255. void zend_test_fiber_init(void)
  256. {
  257. zend_test_fiber_class = register_class__ZendTestFiber();
  258. zend_test_fiber_class->create_object = zend_test_fiber_object_create;
  259. zend_test_fiber_handlers = std_object_handlers;
  260. zend_test_fiber_handlers.dtor_obj = zend_test_fiber_object_destroy;
  261. zend_test_fiber_handlers.free_obj = zend_test_fiber_object_free;
  262. }