| /* Test backtrace and backtrace_symbols for signal frames, where a |
| system call was interrupted by a signal. |
| 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" |
| |
| #ifndef SIGACTION_FLAGS |
| # define SIGACTION_FLAGS 0 |
| #endif |
| |
| 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, read, 3 * fn, and do_test. */ |
| #define NUM_FUNCTIONS 7 |
| |
| void |
| handle_signal (int signum) |
| { |
| void *addresses[NUM_FUNCTIONS]; |
| char **symbols; |
| int n; |
| int i; |
| |
| /* Get the backtrace addresses. */ |
| n = backtrace (addresses, sizeof (addresses) / sizeof (addresses[0])); |
| printf ("Obtained backtrace with %d functions\n", n); |
| /* Check that there are at least seven functions. */ |
| if (n < NUM_FUNCTIONS) |
| { |
| FAIL (); |
| 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 (); |
| return; |
| } |
| /* Do not check name for signal trampoline. */ |
| i = 2; |
| if (!match (symbols[i++], "read")) |
| { |
| /* Perhaps symbols[2] is __kernel_vsyscall? */ |
| if (!match (symbols[i++], "read")) |
| { |
| FAIL (); |
| return; |
| } |
| } |
| for (; 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, int flags) |
| { |
| pid_t parent_pid, child_pid; |
| int pipefd[2]; |
| char r[1]; |
| struct sigaction act; |
| |
| if (c > 0) |
| { |
| fn (c - 1, flags); |
| return x; |
| } |
| |
| memset (&act, 0, sizeof (act)); |
| act.sa_handler = handle_signal; |
| act.sa_flags = flags; |
| sigemptyset (&act.sa_mask); |
| sigaction (SIGUSR1, &act, NULL); |
| parent_pid = getpid (); |
| if (pipe (pipefd) == -1) |
| abort (); |
| |
| 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. */ |
| read (pipefd[0], r, 1); |
| |
| return 0; |
| } |
| |
| NO_INLINE static int |
| do_test (void) |
| { |
| fn (2, SIGACTION_FLAGS); |
| return ret; |
| } |