0b7c7473b9
When debugging failures in this test, it's helpful to see as much output as possible. So rather than returning immediately, let the code run as far as it can. We still mark failures as soon as they happen. Signed-off-by: Mike Frysinger <vapier@gentoo.org>
124 lines
2.9 KiB
C
124 lines
2.9 KiB
C
/* Test backtrace and backtrace_symbols for signal frames.
|
|
Copyright (C) 2011-2014 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library.
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
The GNU C Library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with the GNU C Library; if not, see
|
|
<http://www.gnu.org/licenses/>. */
|
|
|
|
#include <execinfo.h>
|
|
#include <search.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <signal.h>
|
|
#include <unistd.h>
|
|
|
|
#include "tst-backtrace.h"
|
|
|
|
static int do_test (void);
|
|
#define TEST_FUNCTION do_test ()
|
|
#include "../test-skeleton.c"
|
|
|
|
/* The backtrace should include at least handle_signal, a signal
|
|
trampoline, 3 * fn, and do_test. */
|
|
#define NUM_FUNCTIONS 6
|
|
|
|
volatile int sig_handled = 0;
|
|
|
|
void
|
|
handle_signal (int signum)
|
|
{
|
|
void *addresses[NUM_FUNCTIONS];
|
|
char **symbols;
|
|
int n;
|
|
int i;
|
|
|
|
sig_handled = 1;
|
|
|
|
/* Get the backtrace addresses. */
|
|
n = backtrace (addresses, sizeof (addresses) / sizeof (addresses[0]));
|
|
printf ("Obtained backtrace with %d functions (but wanted at least %d)\n",
|
|
n, NUM_FUNCTIONS);
|
|
/* Check that there are at least six functions. */
|
|
if (n < NUM_FUNCTIONS)
|
|
{
|
|
FAIL ();
|
|
/* Only return if we got no symbols at all. The partial output is
|
|
still useful for debugging failures. */
|
|
if (n <= 0)
|
|
return;
|
|
}
|
|
/* Convert them to symbols. */
|
|
symbols = backtrace_symbols (addresses, n);
|
|
/* Check that symbols were obtained. */
|
|
if (symbols == NULL)
|
|
{
|
|
FAIL ();
|
|
return;
|
|
}
|
|
for (i = 0; i < n; ++i)
|
|
printf ("Function %d: %s\n", i, symbols[i]);
|
|
/* Check that the function names obtained are accurate. */
|
|
if (!match (symbols[0], "handle_signal"))
|
|
FAIL ();
|
|
/* Do not check name for signal trampoline. */
|
|
for (i = 2; i < n - 1; i++)
|
|
if (!match (symbols[i], "fn"))
|
|
{
|
|
FAIL ();
|
|
return;
|
|
}
|
|
/* Symbol names are not available for static functions, so we do not
|
|
check do_test. */
|
|
}
|
|
|
|
NO_INLINE int
|
|
fn (int c)
|
|
{
|
|
pid_t parent_pid, child_pid;
|
|
|
|
if (c > 0)
|
|
{
|
|
fn (c - 1);
|
|
return x;
|
|
}
|
|
|
|
signal (SIGUSR1, handle_signal);
|
|
parent_pid = getpid ();
|
|
|
|
child_pid = fork ();
|
|
if (child_pid == (pid_t) -1)
|
|
abort ();
|
|
else if (child_pid == 0)
|
|
{
|
|
sleep (1);
|
|
kill (parent_pid, SIGUSR1);
|
|
_exit (0);
|
|
}
|
|
|
|
/* In the parent. */
|
|
while (sig_handled == 0)
|
|
;
|
|
|
|
return 0;
|
|
}
|
|
|
|
NO_INLINE static int
|
|
do_test (void)
|
|
{
|
|
fn (2);
|
|
return ret;
|
|
}
|