nlr.c 3.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  1. /*
  2. * This file is part of the MicroPython project, http://micropython.org/
  3. *
  4. * The MIT License (MIT)
  5. *
  6. * Copyright (c) 2013-2023 Damien P. George
  7. *
  8. * Permission is hereby granted, free of charge, to any person obtaining a copy
  9. * of this software and associated documentation files (the "Software"), to deal
  10. * in the Software without restriction, including without limitation the rights
  11. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  12. * copies of the Software, and to permit persons to whom the Software is
  13. * furnished to do so, subject to the following conditions:
  14. *
  15. * The above copyright notice and this permission notice shall be included in
  16. * all copies or substantial portions of the Software.
  17. *
  18. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  19. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  20. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  21. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  22. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  23. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  24. * THE SOFTWARE.
  25. */
  26. #include "py/mpstate.h"
  27. #if !MICROPY_NLR_SETJMP
  28. // When not using setjmp, nlr_push_tail is called from inline asm so needs special care
  29. #if MICROPY_NLR_X86 && MICROPY_NLR_OS_WINDOWS
  30. // On these 32-bit platforms make sure nlr_push_tail doesn't have a leading underscore
  31. unsigned int nlr_push_tail(nlr_buf_t *nlr) asm ("nlr_push_tail");
  32. #else
  33. // LTO can't see inside inline asm functions so explicitly mark nlr_push_tail as used
  34. __attribute__((used)) unsigned int nlr_push_tail(nlr_buf_t *nlr);
  35. #endif
  36. #endif
  37. unsigned int nlr_push_tail(nlr_buf_t *nlr) {
  38. nlr_buf_t **top = &MP_STATE_THREAD(nlr_top);
  39. nlr->prev = *top;
  40. MP_NLR_SAVE_PYSTACK(nlr);
  41. *top = nlr;
  42. return 0; // normal return
  43. }
  44. void nlr_pop(void) {
  45. nlr_buf_t **top = &MP_STATE_THREAD(nlr_top);
  46. *top = (*top)->prev;
  47. }
  48. void nlr_push_jump_callback(nlr_jump_callback_node_t *node, nlr_jump_callback_fun_t fun) {
  49. nlr_jump_callback_node_t **top = &MP_STATE_THREAD(nlr_jump_callback_top);
  50. node->prev = *top;
  51. node->fun = fun;
  52. *top = node;
  53. }
  54. void nlr_pop_jump_callback(bool run_callback) {
  55. nlr_jump_callback_node_t **top = &MP_STATE_THREAD(nlr_jump_callback_top);
  56. nlr_jump_callback_node_t *cur = *top;
  57. *top = (*top)->prev;
  58. if (run_callback) {
  59. cur->fun(cur);
  60. }
  61. }
  62. // This function pops and runs all callbacks that were registered after `nlr`
  63. // was pushed (via nlr_push). It assumes:
  64. // - a descending C stack,
  65. // - that all nlr_jump_callback_node_t's in the linked-list pointed to by
  66. // nlr_jump_callback_top are on the C stack
  67. // It works by popping each node in turn until the next node is NULL or above
  68. // the `nlr` pointer on the C stack (and so pushed before `nlr` was pushed).
  69. void nlr_call_jump_callbacks(nlr_buf_t *nlr) {
  70. nlr_jump_callback_node_t **top = &MP_STATE_THREAD(nlr_jump_callback_top);
  71. while (*top != NULL && (void *)*top < (void *)nlr) {
  72. nlr_pop_jump_callback(true);
  73. }
  74. }
  75. #if MICROPY_ENABLE_VM_ABORT
  76. NORETURN void nlr_jump_abort(void) {
  77. MP_STATE_THREAD(nlr_top) = MP_STATE_VM(nlr_abort);
  78. nlr_jump(NULL);
  79. }
  80. #endif