diff options
Diffstat (limited to 'translate/grt/config/win32.c')
-rw-r--r-- | translate/grt/config/win32.c | 48 |
1 files changed, 47 insertions, 1 deletions
diff --git a/translate/grt/config/win32.c b/translate/grt/config/win32.c index 583b885ba..18e5a2d4c 100644 --- a/translate/grt/config/win32.c +++ b/translate/grt/config/win32.c @@ -19,6 +19,8 @@ #include <windows.h> #include <stdio.h> +#include <setjmp.h> +#include <assert.h> struct stack_type { @@ -27,7 +29,8 @@ struct stack_type void *arg; // Function argument. }; -static struct stack_type main_stack_context; +static struct stack_type main_stack_context; +static struct stack_type *current; extern void grt_set_main_stack (struct stack_type *stack); void grt_stack_init(void) @@ -40,6 +43,7 @@ void grt_stack_init(void) abort (); } grt_set_main_stack (&main_stack_context); + current = &main_stack_context; } static VOID __stdcall @@ -71,10 +75,22 @@ grt_stack_create (void (*func)(void *), void *arg) return res; } +static int run_env_en; +static jmp_buf run_env; +static int need_longjmp; + void grt_stack_switch (struct stack_type *to, struct stack_type *from) { + assert (current == from); + current = to; SwitchToFiber (to->fiber); + if (from == &main_stack_context && need_longjmp) + { + /* We returned to do the longjump. */ + current = &main_stack_context; + longjmp (run_env, need_longjmp); + } } void @@ -84,6 +100,36 @@ grt_stack_delete (struct stack_type *stack) stack->fiber = NULL; } +void +__ghdl_maybe_return_via_longjump (int val) +{ + if (!run_env_en) + return; + + if (current != &main_stack_context) + { + /* We are allowed to jump only in the same stack. + First switch back to the main thread. */ + need_longjmp = val; + SwitchToFiber (main_stack_context.fiber); + } + else + longjmp (run_env, val); +} + +int +__ghdl_run_through_longjump (int (*func)(void)) +{ + int res; + + run_env_en = 1; + res = setjmp (run_env); + if (res == 0) + res = (*func)(); + run_env_en = 0; + return res; +} + #ifndef WITH_GNAT_RUN_TIME void __gnat_raise_storage_error(void) { |