[Git][cmucl/cmucl][issue-97-define-ud2-inst] Add sigtrap handler to handle single-stepping traps
Raymond Toy
gitlab at common-lisp.net
Tue Mar 23 02:57:21 UTC 2021
Raymond Toy pushed to branch issue-97-define-ud2-inst at cmucl / cmucl
Commits:
8817798f by Raymond Toy at 2021-03-22T19:57:09-07:00
Add sigtrap handler to handle single-stepping traps
Needed to handle single-stepping. There's also a lot of other changes
used to debug this. They eventually need to be cleaned up.
- - - - -
1 changed file:
- src/lisp/x86-arch.c
Changes:
=====================================
src/lisp/x86-arch.c
=====================================
@@ -142,7 +142,7 @@ arch_skip_instruction(os_context_t * context)
{
int vlen, code;
- DPRINTF(0, (stderr, "[arch_skip_inst at %lx>]\n", SC_PC(context)));
+ DPRINTF(1, (stderr, "[arch_skip_inst at %lx>]\n", SC_PC(context)));
/* Get and skip the lisp error code. */
char* pc = (char *) SC_PC(context);
@@ -206,8 +206,8 @@ arch_set_pseudo_atomic_interrupted(os_context_t * context)
unsigned long
arch_install_breakpoint(void *pc)
{
- char* ptr = (char *) pc;
- unsigned long result = *(unsigned long *) pc;
+ unsigned char* ptr = (unsigned char *) pc;
+ unsigned long result = ptr[0] | (ptr[1] << 8) | (ptr[2] << 16) | (ptr[3] << 24);
fprintf(stderr, "arch_install_breakpoint at %p, old code = 0x%lx\n",
pc, result);
@@ -227,8 +227,13 @@ arch_install_breakpoint(void *pc)
void
arch_remove_breakpoint(void *pc, unsigned long orig_inst)
{
- *((char *) pc) = orig_inst & 0xff;
- *((char *) pc + 1) = (orig_inst & 0xff00) >> 8;
+ fprintf(stderr, "arch_remove_breakpoint: %p orig %lx\n",
+ pc, orig_inst);
+ unsigned char *ptr = (unsigned char *) pc;
+ ptr[0] = orig_inst & 0xff;
+ ptr[1] = (orig_inst >> 8) & 0xff;
+ ptr[2] = (orig_inst >> 16) & 0xff;
+ ptr[3] = (orig_inst >> 24) & 0xff;
}
@@ -249,14 +254,23 @@ unsigned int single_step_save3;
void
arch_do_displaced_inst(os_context_t * context, unsigned long orig_inst)
{
- unsigned int *pc = (unsigned int *) SC_PC(context);
+ unsigned char *pc = (unsigned char *) SC_PC(context);
+ fprintf(stderr, "arch_do_displaced_inst: pc %p orig_inst %lx\n",
+ pc, orig_inst);
+
/*
* Put the original instruction back.
*/
+#if 0
*((char *) pc) = orig_inst & 0xff;
*((char *) pc + 1) = (orig_inst & 0xff00) >> 8;
+#else
+ pc[0] = orig_inst & 0xff;
+ pc[1] = (orig_inst >> 8) & 0xff;
+ pc[2] = (orig_inst >> 16) & 0xff;
+#endif
#ifdef SC_EFLAGS
/* Enable single-stepping */
@@ -319,8 +333,8 @@ sigill_handler(HANDLER_ARGS)
fprintf(stderr, "sigtrap(%d %d %p)\n", signal, CODE(code), os_context);
#endif
- if (single_stepping && (signal == SIGTRAP)) {
-#if 0
+ if (single_stepping && (signal == SIGILL)) {
+#if 1
fprintf(stderr, "* Single step trap %p\n", single_stepping);
#endif
@@ -338,7 +352,9 @@ sigill_handler(HANDLER_ARGS)
/*
* Re-install the breakpoint if possible.
*/
- if ((int) SC_PC(os_context) == (int) single_stepping + 1)
+ fprintf(stderr, "* Reinstall breakpoint at single_stepping %p\n", single_stepping);
+
+ if ((int) SC_PC(os_context) >= (int) single_stepping + 3)
fprintf(stderr, "* Breakpoint not re-install\n");
else {
char *ptr = (char *) single_stepping;
@@ -460,10 +476,70 @@ sigill_handler(HANDLER_ARGS)
}
}
+void
+sigtrap_handler(HANDLER_ARGS)
+{
+ os_context_t* os_context = (os_context_t *) context;
+
+#if 1
+ fprintf(stderr,"sigtrap: fp=%lx sp=%lx pc=%lx { %x, %x, %x, %x, %x }\n",
+ SC_REG(context, reg_FP),
+ SC_REG(context, reg_SP),
+ SC_PC(context),
+ *(unsigned char*)(SC_PC(context) + 0), /* 0x0F */
+ *(unsigned char*)(SC_PC(context) + 1), /* 0x0B */
+ *(unsigned char*)(SC_PC(context) + 2),
+ *(unsigned char*)(SC_PC(context) + 3),
+ *(unsigned char*)(SC_PC(context) + 4));
+#endif
+ if (single_stepping && (signal == SIGTRAP)) {
+#if 1
+ fprintf(stderr, "* Single step trap %p\n", single_stepping);
+#endif
+
+#ifdef SC_EFLAGS
+ /* Disable single-stepping */
+ SC_EFLAGS(os_context) ^= 0x100;
+#else
+ /* Un-install single step helper instructions. */
+ *(single_stepping - 3) = single_step_save1;
+ *(single_stepping - 2) = single_step_save2;
+ *(single_stepping - 1) = single_step_save3;
+ DPRINTF(0, (stderr, "Uninstalling helper instructions\n"));
+#endif
+
+ /*
+ * Re-install the breakpoint if possible.
+ */
+ fprintf(stderr, "* Maybe reinstall breakpoint for pc %p with single_stepping %p\n",
+ (void*) SC_PC(os_context), single_stepping);
+
+ if ((unsigned long) SC_PC(os_context) <= (unsigned long) single_stepping + 3)
+ fprintf(stderr, "* Breakpoint not re-install\n");
+ else {
+ char *ptr = (char *) single_stepping;
+
+#if 0
+ ptr[0] = BREAKPOINT_INST; /* x86 INT3 */
+ ptr[1] = trap_Breakpoint;
+#else
+ ptr[0] = 0x0f;
+ ptr[1] = 0x0b;
+ ptr[2] = trap_Breakpoint;
+#endif
+ }
+
+ single_stepping = NULL;
+ return;
+ }
+}
+
+
void
arch_install_interrupt_handlers(void)
{
interrupt_install_low_level_handler(SIGILL, sigill_handler);
+ interrupt_install_low_level_handler(SIGTRAP, sigtrap_handler);
}
View it on GitLab: https://gitlab.common-lisp.net/cmucl/cmucl/-/commit/8817798f004ea8e745201c5a326de9ec65a3e850
--
View it on GitLab: https://gitlab.common-lisp.net/cmucl/cmucl/-/commit/8817798f004ea8e745201c5a326de9ec65a3e850
You're receiving this email because of your account on gitlab.common-lisp.net.
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://mailman.common-lisp.net/pipermail/cmucl-cvs/attachments/20210323/7e0a652e/attachment-0001.html>
More information about the cmucl-cvs
mailing list