62 lines
2.9 KiB
C
62 lines
2.9 KiB
C
#ifndef COSMOPOLITAN_LIBC_LOG_GDB_H_
|
|
#define COSMOPOLITAN_LIBC_LOG_GDB_H_
|
|
#include "libc/calls/calls.h"
|
|
#include "libc/calls/wait4.h"
|
|
#include "libc/sysv/consts/nr.h"
|
|
#include "libc/sysv/consts/w.h"
|
|
#if !(__ASSEMBLER__ + __LINKER__ + 0)
|
|
COSMOPOLITAN_C_START_
|
|
|
|
/**
|
|
* @fileoverview GDB Attach Support Code.
|
|
*
|
|
* The goal of these macros is to make the backtrace into the failing
|
|
* code as short as possible. It also helps avoid GDB getting confused
|
|
* about how we don't use its readability destroying unwind directives.
|
|
*/
|
|
|
|
extern volatile int g_gdbsync;
|
|
|
|
int gdbexec(const char *);
|
|
int attachdebugger(intptr_t);
|
|
|
|
#define attachdebugger(CONTINUE_TO_ADDR) /* shorten backtraces */ \
|
|
SYNCHRONIZE_DEBUGGER((attachdebugger)(CONTINUE_TO_ADDR))
|
|
|
|
#define SYNCHRONIZE_DEBUGGER(PID) \
|
|
({ \
|
|
int Rc, Pid = (PID); \
|
|
if (Pid != -1) { \
|
|
while ((Rc = __inline_wait4(Pid, NULL, WNOHANG, NULL)) == 0) { \
|
|
if (g_gdbsync) { \
|
|
g_gdbsync = 0; \
|
|
if (Rc > 0) Pid = 0; \
|
|
break; \
|
|
} else { \
|
|
sched_yield(); \
|
|
} \
|
|
} \
|
|
} \
|
|
Pid; \
|
|
})
|
|
|
|
#define __inline_wait4(PID, OPT_OUT_WSTATUS, OPTIONS, OPT_OUT_RUSAGE) \
|
|
({ \
|
|
int64_t WaAx; \
|
|
if (!IsWindows()) { \
|
|
register void *Reg10 asm("r10") = (OPT_OUT_RUSAGE); \
|
|
asm volatile("syscall" \
|
|
: "=a"(WaAx) \
|
|
: "0"(__NR_wait4), "D"(PID), "S"(OPT_OUT_WSTATUS), \
|
|
"d"(OPTIONS), "r"(Reg10) \
|
|
: "rcx", "r11", "cc", "memory"); \
|
|
} else { \
|
|
WaAx = wait4$nt(PID, OPT_OUT_WSTATUS, OPTIONS, OPT_OUT_RUSAGE); \
|
|
} \
|
|
WaAx; \
|
|
})
|
|
|
|
COSMOPOLITAN_C_END_
|
|
#endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */
|
|
#endif /* COSMOPOLITAN_LIBC_LOG_GDB_H_ */
|