summaryrefslogtreecommitdiffstats
path: root/unwind.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2019-04-29 06:48:00 -0700
committerKaz Kylheku <kaz@kylheku.com>2019-04-29 06:48:00 -0700
commit06d0ada8a9ff7078f8ab89d5b4ce36f04587dc62 (patch)
treea1a2ce6b398183fd9a3fff1ef789b6e71cfe4535 /unwind.c
parent37e9a5d1f8ec90c2fe1cddd0546655ad2afc1dac (diff)
downloadtxr-06d0ada8a9ff7078f8ab89d5b4ce36f04587dc62.tar.gz
txr-06d0ada8a9ff7078f8ab89d5b4ce36f04587dc62.tar.bz2
txr-06d0ada8a9ff7078f8ab89d5b4ce36f04587dc62.zip
debugger: expand frames.
This patch adds special unwind frames for backtracing expansions. With this, we can get rid of the global variable last_form_expanded, since to get the last form expanded, we just search for the most enclosing expand frame. * eval.c (last_form_expanded): Global variable removed. (error_trace): Use uw_last_form_expanded() instead of last_form_expanded. (expand_eval): No need to save and restore last_form_expanded any more. (expand_lisp_setq, expand_setqf, expand_lisp1, do_expand): Use uw_last_form_expanded(). (expand, do_macroexpand_1): Push and pop expand frame. This fixes a bug: do_macroexpand_1 was not recording last_form_expanded. Evaluation of top-level forms uses explicit macroexpansion, therefore top-level evaluation was neglecting to set last_form_expanded. This explains weird behavior I saw in the listener from time to time, when errors would report against the expansion of the wrong form. (eval_init): Remove reference to last_form_expanded variable. * eval.h (last_form_expanded): Declaration removed. * share/txr/stdlib/debug.tl (expand-frame print-trace, expand-frame loc): New methods. (print-backtrace): Include uw-expand frames in the backtrace. * unwind.c (expand_frame_type): New static variable. (uw_find_frames_by_mask): Handle UW_EXPAND. (uw_last_form_expanded, uw_push_expand): New functions. (uw_late_init): Register expand-frame struct type. * unwind.h (enum uw_frtype): New enum member, UW_EXPAND. (uw_last_form_expanded, uw_push_expand): Declared.
Diffstat (limited to 'unwind.c')
-rw-r--r--unwind.c40
1 files changed, 39 insertions, 1 deletions
diff --git a/unwind.c b/unwind.c
index 48852dbc..ca8762fd 100644
--- a/unwind.c
+++ b/unwind.c
@@ -67,7 +67,7 @@ static val sys_cont_s, sys_cont_poison_s;
static val sys_cont_free_s, sys_capture_cont_s;
static val frame_type, catch_frame_type, handle_frame_type;
-static val fcall_frame_type, eval_frame_type;
+static val fcall_frame_type, eval_frame_type, expand_frame_type;
static val deferred_warnings, tentative_defs;
@@ -440,6 +440,13 @@ val uw_find_frames_by_mask(val mask_in)
slotset(frame, env_s, fr->el.env);
break;
}
+ case UW_EXPAND:
+ {
+ frame = allocate_struct(expand_frame_type);
+ slotset(frame, form_s, fr->el.form);
+ slotset(frame, env_s, fr->el.env);
+ break;
+ }
default:
break;
}
@@ -454,6 +461,22 @@ val uw_find_frames_by_mask(val mask_in)
#endif
+#if CONFIG_DEBUG_SUPPORT
+
+val uw_last_form_expanded(void)
+{
+ uw_frame_t *fr;
+
+ for (fr = uw_stack; fr != 0; fr = fr->uw.up) {
+ if (fr->uw.type == UW_EXPAND)
+ return fr->el.form;
+ }
+
+ return nil;
+}
+
+#endif
+
val uw_invoke_catch(val catch_frame, val sym, struct args *args)
{
uw_frame_t *ex, *ex_point;
@@ -593,6 +616,16 @@ void uw_push_eval(uw_frame_t *fr, val form, val env)
uw_stack = fr;
}
+void uw_push_expand(uw_frame_t *fr, val form, val env)
+{
+ memset(fr, 0, sizeof *fr);
+ fr->el.type = UW_EXPAND;
+ fr->el.form = form;
+ fr->el.env = env;
+ fr->el.up = uw_stack;
+ uw_stack = fr;
+}
+
#endif
static val exception_subtypes;
@@ -1184,6 +1217,10 @@ void uw_late_init(void)
frame_type, nil,
list(form_s, env_s, nao),
nil, nil, nil, nil);
+ expand_frame_type = make_struct_type(intern(lit("expand-frame"), user_package),
+ frame_type, nil,
+ list(form_s, env_s, nao),
+ nil, nil, nil, nil);
#endif
reg_mac(intern(lit("defex"), user_package), func_n2(me_defex));
reg_var(unhandled_hook_s = intern(lit("*unhandled-hook*"),
@@ -1220,6 +1257,7 @@ void uw_late_init(void)
reg_varl(intern(lit("uw-guard"), system_package), num_fast(1U <<UW_GUARD));
reg_varl(intern(lit("uw-fcall"), system_package), num_fast(1U <<UW_FCALL));
reg_varl(intern(lit("uw-eval"), system_package), num_fast(1U <<UW_EVAL));
+ reg_varl(intern(lit("uw-expand"), system_package), num_fast(1U <<UW_EXPAND));
reg_fun(intern(lit("find-frames-by-mask"), user_package), func_n1(uw_find_frames_by_mask));
#endif
uw_register_subtype(continue_s, restart_s);