1da0c48c4Sopenharmony_ci/* Test program for unwinding of complicated DWARF expressions. 2da0c48c4Sopenharmony_ci Copyright (C) 2013, 2015, 2018 Red Hat, Inc. 3da0c48c4Sopenharmony_ci This file is part of elfutils. 4da0c48c4Sopenharmony_ci 5da0c48c4Sopenharmony_ci This file is free software; you can redistribute it and/or modify 6da0c48c4Sopenharmony_ci it under the terms of the GNU General Public License as published by 7da0c48c4Sopenharmony_ci the Free Software Foundation; either version 3 of the License, or 8da0c48c4Sopenharmony_ci (at your option) any later version. 9da0c48c4Sopenharmony_ci 10da0c48c4Sopenharmony_ci elfutils is distributed in the hope that it will be useful, but 11da0c48c4Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 12da0c48c4Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 13da0c48c4Sopenharmony_ci GNU General Public License for more details. 14da0c48c4Sopenharmony_ci 15da0c48c4Sopenharmony_ci You should have received a copy of the GNU General Public License 16da0c48c4Sopenharmony_ci along with this program. If not, see <http://www.gnu.org/licenses/>. */ 17da0c48c4Sopenharmony_ci 18da0c48c4Sopenharmony_ci#include <config.h> 19da0c48c4Sopenharmony_ci#include <inttypes.h> 20da0c48c4Sopenharmony_ci#include <stdio_ext.h> 21da0c48c4Sopenharmony_ci#include <locale.h> 22da0c48c4Sopenharmony_ci#include <errno.h> 23da0c48c4Sopenharmony_ci#include <unistd.h> 24da0c48c4Sopenharmony_ci#include <sys/types.h> 25da0c48c4Sopenharmony_ci#include ELFUTILS_HEADER(dwfl) 26da0c48c4Sopenharmony_ci#include "system.h" 27da0c48c4Sopenharmony_ci 28da0c48c4Sopenharmony_ci#ifndef __linux__ 29da0c48c4Sopenharmony_ci 30da0c48c4Sopenharmony_ciint 31da0c48c4Sopenharmony_cimain (int argc __attribute__ ((unused)), char **argv) 32da0c48c4Sopenharmony_ci{ 33da0c48c4Sopenharmony_ci fprintf (stderr, "%s: Unwinding not supported for this architecture\n", 34da0c48c4Sopenharmony_ci argv[0]); 35da0c48c4Sopenharmony_ci return 77; 36da0c48c4Sopenharmony_ci} 37da0c48c4Sopenharmony_ci 38da0c48c4Sopenharmony_ci#else /* __linux__ */ 39da0c48c4Sopenharmony_ci#include <sys/ptrace.h> 40da0c48c4Sopenharmony_ci#include <sys/wait.h> 41da0c48c4Sopenharmony_ci#include <signal.h> 42da0c48c4Sopenharmony_ci 43da0c48c4Sopenharmony_ci#define main cleanup_13_main 44da0c48c4Sopenharmony_ci#include "cleanup-13.c" 45da0c48c4Sopenharmony_ci#undef main 46da0c48c4Sopenharmony_ci 47da0c48c4Sopenharmony_cistatic void 48da0c48c4Sopenharmony_cireport_pid (Dwfl *dwfl, pid_t pid) 49da0c48c4Sopenharmony_ci{ 50da0c48c4Sopenharmony_ci int result = dwfl_linux_proc_report (dwfl, pid); 51da0c48c4Sopenharmony_ci if (result < 0) 52da0c48c4Sopenharmony_ci error (2, 0, "dwfl_linux_proc_report: %s", dwfl_errmsg (-1)); 53da0c48c4Sopenharmony_ci else if (result > 0) 54da0c48c4Sopenharmony_ci error (2, result, "dwfl_linux_proc_report"); 55da0c48c4Sopenharmony_ci 56da0c48c4Sopenharmony_ci if (dwfl_report_end (dwfl, NULL, NULL) != 0) 57da0c48c4Sopenharmony_ci error (2, 0, "dwfl_report_end: %s", dwfl_errmsg (-1)); 58da0c48c4Sopenharmony_ci 59da0c48c4Sopenharmony_ci result = dwfl_linux_proc_attach (dwfl, pid, true); 60da0c48c4Sopenharmony_ci if (result < 0) 61da0c48c4Sopenharmony_ci error (2, 0, "dwfl_linux_proc_attach: %s", dwfl_errmsg (-1)); 62da0c48c4Sopenharmony_ci else if (result > 0) 63da0c48c4Sopenharmony_ci error (2, result, "dwfl_linux_proc_attach"); 64da0c48c4Sopenharmony_ci} 65da0c48c4Sopenharmony_ci 66da0c48c4Sopenharmony_cistatic Dwfl * 67da0c48c4Sopenharmony_cipid_to_dwfl (pid_t pid) 68da0c48c4Sopenharmony_ci{ 69da0c48c4Sopenharmony_ci static char *debuginfo_path; 70da0c48c4Sopenharmony_ci static const Dwfl_Callbacks proc_callbacks = 71da0c48c4Sopenharmony_ci { 72da0c48c4Sopenharmony_ci .find_debuginfo = dwfl_standard_find_debuginfo, 73da0c48c4Sopenharmony_ci .debuginfo_path = &debuginfo_path, 74da0c48c4Sopenharmony_ci 75da0c48c4Sopenharmony_ci .find_elf = dwfl_linux_proc_find_elf, 76da0c48c4Sopenharmony_ci }; 77da0c48c4Sopenharmony_ci Dwfl *dwfl = dwfl_begin (&proc_callbacks); 78da0c48c4Sopenharmony_ci if (dwfl == NULL) 79da0c48c4Sopenharmony_ci error (2, 0, "dwfl_begin: %s", dwfl_errmsg (-1)); 80da0c48c4Sopenharmony_ci report_pid (dwfl, pid); 81da0c48c4Sopenharmony_ci return dwfl; 82da0c48c4Sopenharmony_ci} 83da0c48c4Sopenharmony_ci 84da0c48c4Sopenharmony_cistatic int 85da0c48c4Sopenharmony_ciframe_callback (Dwfl_Frame *state, void *frame_arg) 86da0c48c4Sopenharmony_ci{ 87da0c48c4Sopenharmony_ci Dwarf_Addr pc; 88da0c48c4Sopenharmony_ci bool isactivation; 89da0c48c4Sopenharmony_ci if (! dwfl_frame_pc (state, &pc, &isactivation)) 90da0c48c4Sopenharmony_ci { 91da0c48c4Sopenharmony_ci error (0, 0, "%s", dwfl_errmsg (-1)); 92da0c48c4Sopenharmony_ci return DWARF_CB_ABORT; 93da0c48c4Sopenharmony_ci } 94da0c48c4Sopenharmony_ci Dwarf_Addr pc_adjusted = pc - (isactivation ? 0 : 1); 95da0c48c4Sopenharmony_ci 96da0c48c4Sopenharmony_ci /* Get PC->SYMNAME. */ 97da0c48c4Sopenharmony_ci Dwfl_Thread *thread = dwfl_frame_thread (state); 98da0c48c4Sopenharmony_ci Dwfl *dwfl = dwfl_thread_dwfl (thread); 99da0c48c4Sopenharmony_ci Dwfl_Module *mod = dwfl_addrmodule (dwfl, pc_adjusted); 100da0c48c4Sopenharmony_ci const char *symname = NULL; 101da0c48c4Sopenharmony_ci if (mod) 102da0c48c4Sopenharmony_ci symname = dwfl_module_addrname (mod, pc_adjusted); 103da0c48c4Sopenharmony_ci 104da0c48c4Sopenharmony_ci printf ("%#" PRIx64 "\t%s\n", (uint64_t) pc, symname ?: "<null>"); 105da0c48c4Sopenharmony_ci 106da0c48c4Sopenharmony_ci if (symname && (strcmp (symname, "main") == 0 107da0c48c4Sopenharmony_ci || strcmp (symname, ".main") == 0)) 108da0c48c4Sopenharmony_ci { 109da0c48c4Sopenharmony_ci kill (dwfl_pid (dwfl), SIGKILL); 110da0c48c4Sopenharmony_ci exit (0); 111da0c48c4Sopenharmony_ci } 112da0c48c4Sopenharmony_ci 113da0c48c4Sopenharmony_ci return DWARF_CB_OK; 114da0c48c4Sopenharmony_ci} 115da0c48c4Sopenharmony_ci 116da0c48c4Sopenharmony_cistatic int 117da0c48c4Sopenharmony_cithread_callback (Dwfl_Thread *thread, void *thread_arg) 118da0c48c4Sopenharmony_ci{ 119da0c48c4Sopenharmony_ci if (dwfl_thread_getframes (thread, frame_callback, NULL) == -1) 120da0c48c4Sopenharmony_ci error (1, 0, "dwfl_thread_getframes: %s", dwfl_errmsg (-1)); 121da0c48c4Sopenharmony_ci 122da0c48c4Sopenharmony_ci /* frame_callback shall exit (0) on success. */ 123da0c48c4Sopenharmony_ci printf ("dwfl_thread_getframes returned, main not found\n"); 124da0c48c4Sopenharmony_ci return DWARF_CB_ABORT; 125da0c48c4Sopenharmony_ci} 126da0c48c4Sopenharmony_ci 127da0c48c4Sopenharmony_ciint 128da0c48c4Sopenharmony_ci__attribute__((section(".main"))) /* Defeat -freorder-blocks-and-partition */ 129da0c48c4Sopenharmony_cimain (int argc __attribute__ ((unused)), char **argv) 130da0c48c4Sopenharmony_ci{ 131da0c48c4Sopenharmony_ci /* We use no threads here which can interfere with handling a stream. */ 132da0c48c4Sopenharmony_ci __fsetlocking (stdin, FSETLOCKING_BYCALLER); 133da0c48c4Sopenharmony_ci __fsetlocking (stdout, FSETLOCKING_BYCALLER); 134da0c48c4Sopenharmony_ci __fsetlocking (stderr, FSETLOCKING_BYCALLER); 135da0c48c4Sopenharmony_ci 136da0c48c4Sopenharmony_ci /* Set locale. */ 137da0c48c4Sopenharmony_ci (void) setlocale (LC_ALL, ""); 138da0c48c4Sopenharmony_ci 139da0c48c4Sopenharmony_ci elf_version (EV_CURRENT); 140da0c48c4Sopenharmony_ci 141da0c48c4Sopenharmony_ci pid_t pid = fork (); 142da0c48c4Sopenharmony_ci switch (pid) 143da0c48c4Sopenharmony_ci { 144da0c48c4Sopenharmony_ci case -1: 145da0c48c4Sopenharmony_ci perror ("fork failed"); 146da0c48c4Sopenharmony_ci exit (-1); 147da0c48c4Sopenharmony_ci case 0:; 148da0c48c4Sopenharmony_ci long l = ptrace (PTRACE_TRACEME, 0, NULL, NULL); 149da0c48c4Sopenharmony_ci if (l != 0) 150da0c48c4Sopenharmony_ci { 151da0c48c4Sopenharmony_ci perror ("PTRACE_TRACEME failed"); 152da0c48c4Sopenharmony_ci exit (-1); 153da0c48c4Sopenharmony_ci } 154da0c48c4Sopenharmony_ci cleanup_13_main (); 155da0c48c4Sopenharmony_ci printf ("cleanup_13_main returned, impossible...\n"); 156da0c48c4Sopenharmony_ci exit (-1); 157da0c48c4Sopenharmony_ci default: 158da0c48c4Sopenharmony_ci break; 159da0c48c4Sopenharmony_ci } 160da0c48c4Sopenharmony_ci 161da0c48c4Sopenharmony_ci errno = 0; 162da0c48c4Sopenharmony_ci int status; 163da0c48c4Sopenharmony_ci pid_t got = waitpid (pid, &status, 0); 164da0c48c4Sopenharmony_ci if (got != pid) 165da0c48c4Sopenharmony_ci error (1, errno, "waitpid returned %d", got); 166da0c48c4Sopenharmony_ci if (!WIFSTOPPED (status)) 167da0c48c4Sopenharmony_ci error (1, 0, "unexpected wait status %u", status); 168da0c48c4Sopenharmony_ci if (WSTOPSIG (status) != SIGABRT) 169da0c48c4Sopenharmony_ci error (1, 0, "unexpected signal %u", WSTOPSIG (status)); 170da0c48c4Sopenharmony_ci 171da0c48c4Sopenharmony_ci Dwfl *dwfl = pid_to_dwfl (pid); 172da0c48c4Sopenharmony_ci if (dwfl_getthreads (dwfl, thread_callback, NULL) == -1) 173da0c48c4Sopenharmony_ci error (1, 0, "dwfl_getthreads: %s", dwfl_errmsg (-1)); 174da0c48c4Sopenharmony_ci 175da0c48c4Sopenharmony_ci /* There is an exit (0) call if we find the "main" frame, */ 176da0c48c4Sopenharmony_ci printf ("dwfl_getthreads returned, main not found\n"); 177da0c48c4Sopenharmony_ci exit (-1); 178da0c48c4Sopenharmony_ci} 179da0c48c4Sopenharmony_ci 180da0c48c4Sopenharmony_ci#endif /* ! __linux__ */ 181da0c48c4Sopenharmony_ci 182