123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686 |
- #ifdef HAVE_CONFIG_H
- # include <config.h>
- #endif
- #include <alloca.h>
- #include <stdio.h>
- #include <unistdio.h>
- #include "backtrace.h"
- #include "boolean.h"
- #include "control.h"
- #include "debug.h"
- #include "deprecation.h"
- #include "eq.h"
- #include "eval.h"
- #include "fluids.h"
- #include "gsubr.h"
- #include "init.h"
- #include "list.h"
- #include "modules.h"
- #include "numbers.h"
- #include "pairs.h"
- #include "ports.h"
- #include "private-options.h"
- #include "smob.h"
- #include "stackchk.h"
- #include "stacks.h"
- #include "strings.h"
- #include "symbols.h"
- #include "variable.h"
- #include "vm.h"
- #include "throw.h"
- static SCM throw_var;
- static SCM exception_handler_fluid;
- static SCM
- catch (SCM tag, SCM thunk, SCM handler, SCM pre_unwind_handler)
- {
- SCM eh, prompt_tag;
- SCM res;
- scm_thread *t = SCM_I_CURRENT_THREAD;
- scm_t_dynstack *dynstack = &t->dynstack;
- scm_t_dynamic_state *dynamic_state = t->dynamic_state;
- jmp_buf registers;
- jmp_buf *prev_registers;
- ptrdiff_t saved_stack_depth;
- uint8_t *mra = NULL;
- if (!scm_is_eq (tag, SCM_BOOL_T) && !scm_is_symbol (tag))
- scm_wrong_type_arg ("catch", 1, tag);
- if (SCM_UNBNDP (handler))
- handler = SCM_BOOL_F;
- else if (!scm_is_true (scm_procedure_p (handler)))
- scm_wrong_type_arg ("catch", 3, handler);
- if (SCM_UNBNDP (pre_unwind_handler))
- pre_unwind_handler = SCM_BOOL_F;
- else if (!scm_is_true (scm_procedure_p (pre_unwind_handler)))
- scm_wrong_type_arg ("catch", 4, pre_unwind_handler);
- prompt_tag = scm_cons (SCM_INUM0, SCM_EOL);
- eh = scm_c_make_vector (3, SCM_BOOL_F);
- scm_c_vector_set_x (eh, 0, tag);
- scm_c_vector_set_x (eh, 1, prompt_tag);
- scm_c_vector_set_x (eh, 2, pre_unwind_handler);
- prev_registers = t->vm.registers;
- saved_stack_depth = t->vm.stack_top - t->vm.sp;
-
- scm_dynstack_push_prompt (dynstack,
- SCM_F_DYNSTACK_PROMPT_ESCAPE_ONLY,
- prompt_tag,
- t->vm.stack_top - t->vm.fp,
- saved_stack_depth,
- t->vm.ip,
- mra,
- ®isters);
- scm_dynstack_push_fluid (dynstack, exception_handler_fluid, eh,
- dynamic_state);
- if (setjmp (registers))
- {
-
- SCM args;
- t->vm.registers = prev_registers;
- scm_gc_after_nonlocal_exit ();
-
- args = scm_i_prompt_pop_abort_args_x (&t->vm, saved_stack_depth);
-
- args = scm_cdr (args);
- return scm_apply_0 (handler, args);
- }
- res = scm_call_0 (thunk);
- scm_dynstack_unwind_fluid (dynstack, dynamic_state);
- scm_dynstack_pop (dynstack);
- return res;
- }
- static void
- default_exception_handler (SCM k, SCM args)
- {
- static int error_printing_error = 0;
- static int error_printing_fallback = 0;
- if (error_printing_fallback)
- fprintf (stderr, "\nFailed to print exception.\n");
- else if (error_printing_error)
- {
- fprintf (stderr, "\nError while printing exception:\n");
- error_printing_fallback = 1;
- fprintf (stderr, "Key: ");
- scm_write (k, scm_current_error_port ());
- fprintf (stderr, ", args: ");
- scm_write (args, scm_current_error_port ());
- scm_newline (scm_current_error_port ());
- }
- else
- {
- fprintf (stderr, "Uncaught exception:\n");
- error_printing_error = 1;
- scm_handle_by_message (NULL, k, args);
- }
-
- fprintf (stderr, "Aborting.\n");
- abort ();
- }
- static void
- abort_to_prompt (SCM prompt_tag, SCM tag, SCM args)
- {
- SCM *tag_and_argv;
- size_t i;
- long n;
- n = scm_ilength (args) + 2;
- tag_and_argv = alloca (sizeof (SCM)*n);
- tag_and_argv[0] = prompt_tag;
- tag_and_argv[1] = tag;
- for (i = 2; i < n; i++, args = scm_cdr (args))
- tag_and_argv[i] = scm_car (args);
- scm_i_vm_emergency_abort (tag_and_argv, n);
-
- abort ();
- }
- static SCM
- throw_without_pre_unwind (SCM tag, SCM args)
- {
- size_t depth = 0;
-
- while (1)
- {
- SCM eh, catch_key, prompt_tag;
- eh = scm_fluid_ref_star (exception_handler_fluid,
- scm_from_size_t (depth++));
- if (scm_is_false (eh))
- break;
- catch_key = scm_c_vector_ref (eh, 0);
- if (!scm_is_eq (catch_key, SCM_BOOL_T) && !scm_is_eq (catch_key, tag))
- continue;
- if (scm_is_true (scm_c_vector_ref (eh, 2)))
- {
- const char *key_chars;
- if (scm_i_is_narrow_symbol (tag))
- key_chars = scm_i_symbol_chars (tag);
- else
- key_chars = "(wide symbol)";
- fprintf (stderr, "Warning: Unwind-only `%s' exception; "
- "skipping pre-unwind handler.\n", key_chars);
- }
- prompt_tag = scm_c_vector_ref (eh, 1);
- if (scm_is_true (prompt_tag))
- abort_to_prompt (prompt_tag, tag, args);
- }
- default_exception_handler (tag, args);
- return SCM_UNSPECIFIED;
- }
- SCM
- scm_catch (SCM key, SCM thunk, SCM handler)
- {
- return catch (key, thunk, handler, SCM_UNDEFINED);
- }
- SCM
- scm_catch_with_pre_unwind_handler (SCM key, SCM thunk, SCM handler,
- SCM pre_unwind_handler)
- {
- return catch (key, thunk, handler, pre_unwind_handler);
- }
- SCM
- scm_with_throw_handler (SCM key, SCM thunk, SCM handler)
- {
- return catch (key, thunk, SCM_UNDEFINED, handler);
- }
- SCM
- scm_throw (SCM key, SCM args)
- {
- scm_apply_1 (scm_variable_ref (throw_var), key, args);
-
- abort ();
- }
- static scm_t_bits tc16_catch_closure;
- enum {
- CATCH_CLOSURE_BODY,
- CATCH_CLOSURE_HANDLER
- };
- SCM
- scm_i_make_catch_body_closure (scm_t_catch_body body, void *body_data)
- {
- SCM ret;
- SCM_NEWSMOB2 (ret, tc16_catch_closure, body, body_data);
- SCM_SET_SMOB_FLAGS (ret, CATCH_CLOSURE_BODY);
- return ret;
- }
- SCM
- scm_i_make_catch_handler_closure (scm_t_catch_handler handler,
- void *handler_data)
- {
- SCM ret;
- SCM_NEWSMOB2 (ret, tc16_catch_closure, handler, handler_data);
- SCM_SET_SMOB_FLAGS (ret, CATCH_CLOSURE_HANDLER);
- return ret;
- }
- static SCM
- apply_catch_closure (SCM clo, SCM args)
- {
- void *data = (void*)SCM_SMOB_DATA_2 (clo);
- switch (SCM_SMOB_FLAGS (clo))
- {
- case CATCH_CLOSURE_BODY:
- {
- scm_t_catch_body body = (void*)SCM_SMOB_DATA (clo);
- return body (data);
- }
- case CATCH_CLOSURE_HANDLER:
- {
- scm_t_catch_handler handler = (void*)SCM_SMOB_DATA (clo);
- return handler (data, scm_car (args), scm_cdr (args));
- }
- default:
- abort ();
- }
- }
- SCM
- scm_c_catch (SCM tag,
- scm_t_catch_body body, void *body_data,
- scm_t_catch_handler handler, void *handler_data,
- scm_t_catch_handler pre_unwind_handler, void *pre_unwind_handler_data)
- {
- SCM sbody, shandler, spre_unwind_handler;
-
- sbody = scm_i_make_catch_body_closure (body, body_data);
- shandler = scm_i_make_catch_handler_closure (handler, handler_data);
- if (pre_unwind_handler)
- spre_unwind_handler =
- scm_i_make_catch_handler_closure (pre_unwind_handler,
- pre_unwind_handler_data);
- else
- spre_unwind_handler = SCM_UNDEFINED;
-
- return scm_catch_with_pre_unwind_handler (tag, sbody, shandler,
- spre_unwind_handler);
- }
- SCM
- scm_internal_catch (SCM tag,
- scm_t_catch_body body, void *body_data,
- scm_t_catch_handler handler, void *handler_data)
- {
- return scm_c_catch (tag,
- body, body_data,
- handler, handler_data,
- NULL, NULL);
- }
- SCM
- scm_c_with_throw_handler (SCM tag,
- scm_t_catch_body body,
- void *body_data,
- scm_t_catch_handler handler,
- void *handler_data,
- int lazy_catch_p)
- {
- SCM sbody, shandler;
- if (lazy_catch_p)
- scm_c_issue_deprecation_warning
- ("The LAZY_CATCH_P argument to `scm_c_with_throw_handler' is no longer.\n"
- "supported. Instead the handler will be invoked from within the dynamic\n"
- "context of the corresponding `throw'.\n"
- "\nTHIS COULD CHANGE YOUR PROGRAM'S BEHAVIOR.\n\n"
- "Please modify your program to pass 0 as the LAZY_CATCH_P argument,\n"
- "and adapt it (if necessary) to expect to be within the dynamic context\n"
- "of the throw.");
- sbody = scm_i_make_catch_body_closure (body, body_data);
- shandler = scm_i_make_catch_handler_closure (handler, handler_data);
-
- return scm_with_throw_handler (tag, sbody, shandler);
- }
- SCM
- scm_body_thunk (void *body_data)
- {
- struct scm_body_thunk_data *c = (struct scm_body_thunk_data *) body_data;
- return scm_call_0 (c->body_proc);
- }
- SCM
- scm_handle_by_proc (void *handler_data, SCM tag, SCM throw_args)
- {
- SCM *handler_proc_p = (SCM *) handler_data;
- return scm_apply_1 (*handler_proc_p, tag, throw_args);
- }
- struct hbpca_data {
- SCM proc;
- SCM args;
- };
- static SCM
- hbpca_body (void *body_data)
- {
- struct hbpca_data *data = (struct hbpca_data *)body_data;
- return scm_apply_0 (data->proc, data->args);
- }
- SCM
- scm_handle_by_proc_catching_all (void *handler_data, SCM tag, SCM throw_args)
- {
- SCM *handler_proc_p = (SCM *) handler_data;
- struct hbpca_data data;
- data.proc = *handler_proc_p;
- data.args = scm_cons (tag, throw_args);
- return scm_internal_catch (SCM_BOOL_T,
- hbpca_body, &data,
- scm_handle_by_message_noexit, NULL);
- }
- int
- scm_exit_status (SCM args)
- {
- if (scm_is_pair (args))
- {
- SCM cqa = SCM_CAR (args);
-
- if (scm_is_integer (cqa))
- return (scm_to_int (cqa));
- else if (scm_is_false (cqa))
- return EXIT_FAILURE;
- else
- return EXIT_SUCCESS;
- }
- else if (scm_is_null (args))
- return EXIT_SUCCESS;
- else
-
- return EXIT_FAILURE;
- }
-
- static int
- should_print_backtrace (SCM tag, SCM stack)
- {
- return SCM_BACKTRACE_P
- && scm_is_true (stack)
- && scm_initialized_p
-
- && !scm_is_eq (tag, scm_from_latin1_symbol ("read-error"))
- && !scm_is_eq (tag, scm_from_latin1_symbol ("syntax-error"));
- }
- static void
- handler_message (void *handler_data, SCM tag, SCM args)
- {
- SCM p, stack, frame;
- p = scm_current_error_port ();
-
- stack = scm_make_stack (SCM_BOOL_T, scm_list_1 (scm_from_int (2)));
- frame = scm_is_true (stack) ? scm_stack_ref (stack, SCM_INUM0) : SCM_BOOL_F;
- if (should_print_backtrace (tag, stack))
- {
- scm_puts ("Backtrace:\n", p);
- scm_display_backtrace_with_highlights (stack, p,
- SCM_BOOL_F, SCM_BOOL_F,
- SCM_EOL);
- scm_newline (p);
- }
- scm_print_exception (p, frame, tag, args);
- }
- SCM
- scm_handle_by_message (void *handler_data, SCM tag, SCM args)
- {
- if (scm_is_true (scm_eq_p (tag, scm_from_latin1_symbol ("quit"))))
- exit (scm_exit_status (args));
- handler_message (handler_data, tag, args);
- scm_i_pthread_exit (NULL);
-
- return SCM_BOOL_F;
- }
- SCM
- scm_handle_by_message_noexit (void *handler_data, SCM tag, SCM args)
- {
- if (scm_is_true (scm_eq_p (tag, scm_from_latin1_symbol ("quit"))))
- exit (scm_exit_status (args));
- handler_message (handler_data, tag, args);
- return SCM_BOOL_F;
- }
- SCM
- scm_handle_by_throw (void *handler_data SCM_UNUSED, SCM tag, SCM args)
- {
- scm_ithrow (tag, args, 1);
- return SCM_UNSPECIFIED;
- }
- SCM
- scm_ithrow (SCM key, SCM args, int no_return SCM_UNUSED)
- {
- scm_throw (key, args);
- }
- SCM_SYMBOL (scm_stack_overflow_key, "stack-overflow");
- SCM_SYMBOL (scm_out_of_memory_key, "out-of-memory");
- static SCM stack_overflow_args = SCM_BOOL_F;
- static SCM out_of_memory_args = SCM_BOOL_F;
- void
- scm_report_stack_overflow (void)
- {
- if (scm_is_false (stack_overflow_args))
- abort ();
- throw_without_pre_unwind (scm_stack_overflow_key, stack_overflow_args);
-
- abort ();
- }
- void
- scm_report_out_of_memory (void)
- {
- if (scm_is_false (out_of_memory_args))
- abort ();
- throw_without_pre_unwind (scm_out_of_memory_key, out_of_memory_args);
-
- abort ();
- }
- void
- scm_init_throw ()
- {
- tc16_catch_closure = scm_make_smob_type ("catch-closure", 0);
- scm_set_smob_apply (tc16_catch_closure, apply_catch_closure, 0, 0, 1);
- exception_handler_fluid = scm_make_thread_local_fluid (SCM_BOOL_F);
-
- scm_c_define ("%exception-handler", exception_handler_fluid);
- scm_c_define ("catch", scm_c_make_gsubr ("catch", 3, 1, 0, catch));
- throw_var = scm_c_define ("throw", scm_c_make_gsubr ("throw", 1, 0, 1,
- throw_without_pre_unwind));
-
- stack_overflow_args = scm_list_4 (SCM_BOOL_F,
- scm_from_latin1_string ("Stack overflow"),
- SCM_BOOL_F,
- SCM_BOOL_F);
- out_of_memory_args = scm_list_4 (SCM_BOOL_F,
- scm_from_latin1_string ("Out of memory"),
- SCM_BOOL_F,
- SCM_BOOL_F);
- #include "throw.x"
- }
|