mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-13 07:56:17 +01:00
gbacktrace: on macOS support and default to LLDB
Tweaked by Philip Withnall <withnall@endlessm.com> to fix review issues. Closes: #1004
This commit is contained in:
parent
9dc006cfc8
commit
1fbd185430
@ -72,6 +72,17 @@
|
|||||||
static void stack_trace (const char * const *args);
|
static void stack_trace (const char * const *args);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Default to using LLDB for backtraces on macOS. */
|
||||||
|
#ifdef __APPLE__
|
||||||
|
#define USE_LLDB
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef USE_LLDB
|
||||||
|
#define DEBUGGER "lldb"
|
||||||
|
#else
|
||||||
|
#define DEBUGGER "gdb"
|
||||||
|
#endif
|
||||||
|
|
||||||
/* People want to hit this from their debugger... */
|
/* People want to hit this from their debugger... */
|
||||||
GLIB_AVAILABLE_IN_ALL volatile gboolean glib_on_error_halt;
|
GLIB_AVAILABLE_IN_ALL volatile gboolean glib_on_error_halt;
|
||||||
volatile gboolean glib_on_error_halt = TRUE;
|
volatile gboolean glib_on_error_halt = TRUE;
|
||||||
@ -231,7 +242,7 @@ g_on_error_stack_trace (const gchar *prg_name)
|
|||||||
#if defined(G_OS_UNIX)
|
#if defined(G_OS_UNIX)
|
||||||
pid_t pid;
|
pid_t pid;
|
||||||
gchar buf[16];
|
gchar buf[16];
|
||||||
const gchar *args[4] = { "gdb", NULL, NULL, NULL };
|
const gchar *args[5] = { DEBUGGER, NULL, NULL, NULL, NULL };
|
||||||
int status;
|
int status;
|
||||||
|
|
||||||
if (!prg_name)
|
if (!prg_name)
|
||||||
@ -239,8 +250,14 @@ g_on_error_stack_trace (const gchar *prg_name)
|
|||||||
|
|
||||||
_g_sprintf (buf, "%u", (guint) getpid ());
|
_g_sprintf (buf, "%u", (guint) getpid ());
|
||||||
|
|
||||||
|
#ifdef USE_LLDB
|
||||||
|
args[1] = prg_name;
|
||||||
|
args[2] = "-p";
|
||||||
|
args[3] = buf;
|
||||||
|
#else
|
||||||
args[1] = prg_name;
|
args[1] = prg_name;
|
||||||
args[2] = buf;
|
args[2] = buf;
|
||||||
|
#endif
|
||||||
|
|
||||||
pid = fork ();
|
pid = fork ();
|
||||||
if (pid == 0)
|
if (pid == 0)
|
||||||
@ -250,11 +267,19 @@ g_on_error_stack_trace (const gchar *prg_name)
|
|||||||
}
|
}
|
||||||
else if (pid == (pid_t) -1)
|
else if (pid == (pid_t) -1)
|
||||||
{
|
{
|
||||||
perror ("unable to fork gdb");
|
perror ("unable to fork " DEBUGGER);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
waitpid (pid, &status, 0);
|
/* Wait until the child really terminates. On Mac OS X waitpid ()
|
||||||
|
* will also return when the child is being stopped due to tracing.
|
||||||
|
*/
|
||||||
|
while (1)
|
||||||
|
{
|
||||||
|
pid_t retval = waitpid (pid, &status, 0);
|
||||||
|
if (WIFEXITED (retval) || WIFSIGNALED (retval))
|
||||||
|
break;
|
||||||
|
}
|
||||||
#else
|
#else
|
||||||
if (IsDebuggerPresent ())
|
if (IsDebuggerPresent ())
|
||||||
G_BREAKPOINT ();
|
G_BREAKPOINT ();
|
||||||
@ -273,6 +298,8 @@ stack_trace_sigchld (int signum)
|
|||||||
stack_trace_done = TRUE;
|
stack_trace_done = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define BUFSIZE 1024
|
||||||
|
|
||||||
static void
|
static void
|
||||||
stack_trace (const char * const *args)
|
stack_trace (const char * const *args)
|
||||||
{
|
{
|
||||||
@ -282,8 +309,8 @@ stack_trace (const char * const *args)
|
|||||||
fd_set fdset;
|
fd_set fdset;
|
||||||
fd_set readset;
|
fd_set readset;
|
||||||
struct timeval tv;
|
struct timeval tv;
|
||||||
int sel, idx, state;
|
int sel, idx, state, line_idx;
|
||||||
char buffer[256];
|
char buffer[BUFSIZE];
|
||||||
char c;
|
char c;
|
||||||
|
|
||||||
stack_trace_done = FALSE;
|
stack_trace_done = FALSE;
|
||||||
@ -315,7 +342,7 @@ stack_trace (const char * const *args)
|
|||||||
close (2);
|
close (2);
|
||||||
dup (old_err);
|
dup (old_err);
|
||||||
}
|
}
|
||||||
perror ("exec gdb failed");
|
perror ("exec " DEBUGGER " failed");
|
||||||
_exit (0);
|
_exit (0);
|
||||||
}
|
}
|
||||||
else if (pid == (pid_t) -1)
|
else if (pid == (pid_t) -1)
|
||||||
@ -327,11 +354,19 @@ stack_trace (const char * const *args)
|
|||||||
FD_ZERO (&fdset);
|
FD_ZERO (&fdset);
|
||||||
FD_SET (out_fd[0], &fdset);
|
FD_SET (out_fd[0], &fdset);
|
||||||
|
|
||||||
|
#ifdef USE_LLDB
|
||||||
|
write (in_fd[1], "bt\n", 3);
|
||||||
|
write (in_fd[1], "p x = 0\n", 8);
|
||||||
|
write (in_fd[1], "process detach\n", 15);
|
||||||
|
write (in_fd[1], "quit\n", 5);
|
||||||
|
#else
|
||||||
write (in_fd[1], "backtrace\n", 10);
|
write (in_fd[1], "backtrace\n", 10);
|
||||||
write (in_fd[1], "p x = 0\n", 8);
|
write (in_fd[1], "p x = 0\n", 8);
|
||||||
write (in_fd[1], "quit\n", 5);
|
write (in_fd[1], "quit\n", 5);
|
||||||
|
#endif
|
||||||
|
|
||||||
idx = 0;
|
idx = 0;
|
||||||
|
line_idx = 0;
|
||||||
state = 0;
|
state = 0;
|
||||||
|
|
||||||
while (1)
|
while (1)
|
||||||
@ -348,10 +383,15 @@ stack_trace (const char * const *args)
|
|||||||
{
|
{
|
||||||
if (read (out_fd[0], &c, 1))
|
if (read (out_fd[0], &c, 1))
|
||||||
{
|
{
|
||||||
|
line_idx += 1;
|
||||||
switch (state)
|
switch (state)
|
||||||
{
|
{
|
||||||
case 0:
|
case 0:
|
||||||
|
#ifdef USE_LLDB
|
||||||
|
if (c == '*' || (c == ' ' && line_idx == 1))
|
||||||
|
#else
|
||||||
if (c == '#')
|
if (c == '#')
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
state = 1;
|
state = 1;
|
||||||
idx = 0;
|
idx = 0;
|
||||||
@ -359,6 +399,7 @@ stack_trace (const char * const *args)
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 1:
|
case 1:
|
||||||
|
if (idx < BUFSIZE)
|
||||||
buffer[idx++] = c;
|
buffer[idx++] = c;
|
||||||
if ((c == '\n') || (c == '\r'))
|
if ((c == '\n') || (c == '\r'))
|
||||||
{
|
{
|
||||||
@ -366,6 +407,7 @@ stack_trace (const char * const *args)
|
|||||||
_g_fprintf (stdout, "%s", buffer);
|
_g_fprintf (stdout, "%s", buffer);
|
||||||
state = 0;
|
state = 0;
|
||||||
idx = 0;
|
idx = 0;
|
||||||
|
line_idx = 0;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
Loading…
Reference in New Issue
Block a user