|
|
06486d |
From 56e7f58908997b6cfd52d44bca208ef50d1bdf61 Mon Sep 17 00:00:00 2001
|
|
|
06486d |
From: Jakub Filak <jfilak@redhat.com>
|
|
|
06486d |
Date: Thu, 13 Nov 2014 12:08:07 +0100
|
|
|
06486d |
Subject: [ABRT PATCH 77/78] gdb: make gdb aware of the abrt's debuginfo dir
|
|
|
06486d |
|
|
|
06486d |
A debuginfo package might ship an auto-loaded gdb script. If abrt
|
|
|
06486d |
unpacks that package into the abrt's debuginfo cache dir and points gdb
|
|
|
06486d |
to that directory, gdb refuses to auto-loaded that gdb scripts and
|
|
|
06486d |
produces a plenty of warning messages that abrt writes to 'backtrace'
|
|
|
06486d |
file.
|
|
|
06486d |
|
|
|
06486d |
The previous solution of this issue was to turn auto-load off completely
|
|
|
06486d |
but it turned the pretty printer off too.
|
|
|
06486d |
|
|
|
06486d |
The correct solution is to add the abrt's debuginfo cache directory to
|
|
|
06486d |
auto-load safe-path and auto-load scripts-dir settings.
|
|
|
06486d |
|
|
|
06486d |
Thanks Jan Kratochvil <jkratoch@redhat.com>
|
|
|
06486d |
|
|
|
06486d |
Requires: rhbz#1163339
|
|
|
06486d |
Resolves: rhbz#1128637
|
|
|
06486d |
|
|
|
06486d |
Signed-off-by: Jakub Filak <jfilak@redhat.com>
|
|
|
06486d |
---
|
|
|
06486d |
src/lib/hooklib.c | 83 ++++++++++++++++++++++++++++++++++++-------------------
|
|
|
06486d |
1 file changed, 55 insertions(+), 28 deletions(-)
|
|
|
06486d |
|
|
|
06486d |
diff --git a/src/lib/hooklib.c b/src/lib/hooklib.c
|
|
|
06486d |
index 4a50727..1d45cdd 100644
|
|
|
06486d |
--- a/src/lib/hooklib.c
|
|
|
06486d |
+++ b/src/lib/hooklib.c
|
|
|
06486d |
@@ -252,11 +252,12 @@ char *get_backtrace(const char *dump_dir_name, unsigned timeout_sec, const char
|
|
|
06486d |
/* Let user know what's going on */
|
|
|
06486d |
log(_("Generating backtrace"));
|
|
|
06486d |
|
|
|
06486d |
- char *args[21];
|
|
|
06486d |
- args[0] = (char*)"gdb";
|
|
|
06486d |
- args[1] = (char*)"-batch";
|
|
|
06486d |
- args[2] = (char*)"-ex";
|
|
|
06486d |
+ unsigned i = 0;
|
|
|
06486d |
+ char *args[25];
|
|
|
06486d |
+ args[i++] = (char*)"gdb";
|
|
|
06486d |
+ args[i++] = (char*)"-batch";
|
|
|
06486d |
struct strbuf *set_debug_file_directory = strbuf_new();
|
|
|
06486d |
+ unsigned auto_load_base_index = 0;
|
|
|
06486d |
if(debuginfo_dirs == NULL)
|
|
|
06486d |
{
|
|
|
06486d |
// set non-existent debug file directory to prevent resolving
|
|
|
06486d |
@@ -266,6 +267,8 @@ char *get_backtrace(const char *dump_dir_name, unsigned timeout_sec, const char
|
|
|
06486d |
else
|
|
|
06486d |
{
|
|
|
06486d |
strbuf_append_str(set_debug_file_directory, "set debug-file-directory /usr/lib/debug");
|
|
|
06486d |
+
|
|
|
06486d |
+ struct strbuf *debug_directories = strbuf_new();
|
|
|
06486d |
const char *p = debuginfo_dirs;
|
|
|
06486d |
while (1)
|
|
|
06486d |
{
|
|
|
06486d |
@@ -274,11 +277,25 @@ char *get_backtrace(const char *dump_dir_name, unsigned timeout_sec, const char
|
|
|
06486d |
if (*p == '\0')
|
|
|
06486d |
break;
|
|
|
06486d |
const char *colon_or_nul = strchrnul(p, ':');
|
|
|
06486d |
- strbuf_append_strf(set_debug_file_directory, ":%.*s/usr/lib/debug", (int)(colon_or_nul - p), p);
|
|
|
06486d |
+ strbuf_append_strf(debug_directories, "%s%.*s/usr/lib/debug", (debug_directories->len == 0 ? "" : ":"),
|
|
|
06486d |
+ (int)(colon_or_nul - p), p);
|
|
|
06486d |
p = colon_or_nul;
|
|
|
06486d |
}
|
|
|
06486d |
+
|
|
|
06486d |
+ strbuf_append_strf(set_debug_file_directory, ":%s", debug_directories->buf);
|
|
|
06486d |
+
|
|
|
06486d |
+ args[i++] = (char*)"-iex";
|
|
|
06486d |
+ auto_load_base_index = i;
|
|
|
06486d |
+ args[i++] = xasprintf("add-auto-load-safe-path %s", debug_directories->buf);
|
|
|
06486d |
+ args[i++] = (char*)"-iex";
|
|
|
06486d |
+ args[i++] = xasprintf("add-auto-load-scripts-directory %s", debug_directories->buf);
|
|
|
06486d |
+
|
|
|
06486d |
+ strbuf_free(debug_directories);
|
|
|
06486d |
}
|
|
|
06486d |
- args[3] = strbuf_free_nobuf(set_debug_file_directory);
|
|
|
06486d |
+
|
|
|
06486d |
+ args[i++] = (char*)"-ex";
|
|
|
06486d |
+ const unsigned debug_dir_cmd_index = i++;
|
|
|
06486d |
+ args[debug_dir_cmd_index] = strbuf_free_nobuf(set_debug_file_directory);
|
|
|
06486d |
|
|
|
06486d |
/* "file BINARY_FILE" is needed, without it gdb cannot properly
|
|
|
06486d |
* unwind the stack. Currently the unwind information is located
|
|
|
06486d |
@@ -300,27 +317,31 @@ char *get_backtrace(const char *dump_dir_name, unsigned timeout_sec, const char
|
|
|
06486d |
* TODO: check mtimes on COREFILE and BINARY_FILE and not supply
|
|
|
06486d |
* BINARY_FILE if it is newer (to at least avoid gdb complaining).
|
|
|
06486d |
*/
|
|
|
06486d |
- args[4] = (char*)"-ex";
|
|
|
06486d |
- args[5] = xasprintf("file %s", executable);
|
|
|
06486d |
+ args[i++] = (char*)"-ex";
|
|
|
06486d |
+ const unsigned file_cmd_index = i++;
|
|
|
06486d |
+ args[file_cmd_index] = xasprintf("file %s", executable);
|
|
|
06486d |
free(executable);
|
|
|
06486d |
|
|
|
06486d |
- args[6] = (char*)"-ex";
|
|
|
06486d |
- args[7] = xasprintf("core-file %s/"FILENAME_COREDUMP, dump_dir_name);
|
|
|
06486d |
+ args[i++] = (char*)"-ex";
|
|
|
06486d |
+ const unsigned core_cmd_index = i++;
|
|
|
06486d |
+ args[core_cmd_index] = xasprintf("core-file %s/"FILENAME_COREDUMP, dump_dir_name);
|
|
|
06486d |
|
|
|
06486d |
- args[8] = (char*)"-ex";
|
|
|
06486d |
+ args[i++] = (char*)"-ex";
|
|
|
06486d |
+ const unsigned bt_cmd_index = i++;
|
|
|
06486d |
/*args[9] = ... see below */
|
|
|
06486d |
- args[10] = (char*)"-ex";
|
|
|
06486d |
- args[11] = (char*)"info sharedlib";
|
|
|
06486d |
+ args[i++] = (char*)"-ex";
|
|
|
06486d |
+ args[i++] = (char*)"info sharedlib";
|
|
|
06486d |
/* glibc's abort() stores its message in __abort_msg variable */
|
|
|
06486d |
- args[12] = (char*)"-ex";
|
|
|
06486d |
- args[13] = (char*)"print (char*)__abort_msg";
|
|
|
06486d |
- args[14] = (char*)"-ex";
|
|
|
06486d |
- args[15] = (char*)"print (char*)__glib_assert_msg";
|
|
|
06486d |
- args[16] = (char*)"-ex";
|
|
|
06486d |
- args[17] = (char*)"info all-registers";
|
|
|
06486d |
- args[18] = (char*)"-ex";
|
|
|
06486d |
- args[19] = (char*)"disassemble";
|
|
|
06486d |
- args[20] = NULL;
|
|
|
06486d |
+ args[i++] = (char*)"-ex";
|
|
|
06486d |
+ args[i++] = (char*)"print (char*)__abort_msg";
|
|
|
06486d |
+ args[i++] = (char*)"-ex";
|
|
|
06486d |
+ args[i++] = (char*)"print (char*)__glib_assert_msg";
|
|
|
06486d |
+ args[i++] = (char*)"-ex";
|
|
|
06486d |
+ args[i++] = (char*)"info all-registers";
|
|
|
06486d |
+ args[i++] = (char*)"-ex";
|
|
|
06486d |
+ const unsigned dis_cmd_index = i++;
|
|
|
06486d |
+ args[dis_cmd_index] = (char*)"disassemble";
|
|
|
06486d |
+ args[i++] = NULL;
|
|
|
06486d |
|
|
|
06486d |
/* Get the backtrace, but try to cap its size */
|
|
|
06486d |
/* Limit bt depth. With no limit, gdb sometimes OOMs the machine */
|
|
|
06486d |
@@ -330,9 +351,9 @@ char *get_backtrace(const char *dump_dir_name, unsigned timeout_sec, const char
|
|
|
06486d |
char *bt = NULL;
|
|
|
06486d |
while (1)
|
|
|
06486d |
{
|
|
|
06486d |
- args[9] = xasprintf("%s backtrace %u%s", thread_apply_all, bt_depth, full);
|
|
|
06486d |
+ args[bt_cmd_index] = xasprintf("%s backtrace %u%s", thread_apply_all, bt_depth, full);
|
|
|
06486d |
bt = exec_vp(args, /*redirect_stderr:*/ 1, timeout_sec, NULL);
|
|
|
06486d |
- free(args[9]);
|
|
|
06486d |
+ free(args[bt_cmd_index]);
|
|
|
06486d |
if ((bt && strnlen(bt, 256*1024) < 256*1024) || bt_depth <= 32)
|
|
|
06486d |
{
|
|
|
06486d |
break;
|
|
|
06486d |
@@ -357,7 +378,7 @@ char *get_backtrace(const char *dump_dir_name, unsigned timeout_sec, const char
|
|
|
06486d |
* End of assembler dump.
|
|
|
06486d |
* (IOW: "empty" dump)
|
|
|
06486d |
*/
|
|
|
06486d |
- args[19] = (char*)"disassemble $pc-20, $pc+64";
|
|
|
06486d |
+ args[dis_cmd_index] = (char*)"disassemble $pc-20, $pc+64";
|
|
|
06486d |
|
|
|
06486d |
if (bt_depth <= 64 && thread_apply_all[0] != '\0')
|
|
|
06486d |
{
|
|
|
06486d |
@@ -373,9 +394,15 @@ char *get_backtrace(const char *dump_dir_name, unsigned timeout_sec, const char
|
|
|
06486d |
}
|
|
|
06486d |
}
|
|
|
06486d |
|
|
|
06486d |
- free(args[3]);
|
|
|
06486d |
- free(args[5]);
|
|
|
06486d |
- free(args[7]);
|
|
|
06486d |
+ if (auto_load_base_index > 0)
|
|
|
06486d |
+ {
|
|
|
06486d |
+ free(args[auto_load_base_index]);
|
|
|
06486d |
+ free(args[auto_load_base_index + 2]);
|
|
|
06486d |
+ }
|
|
|
06486d |
+
|
|
|
06486d |
+ free(args[debug_dir_cmd_index]);
|
|
|
06486d |
+ free(args[file_cmd_index]);
|
|
|
06486d |
+ free(args[core_cmd_index]);
|
|
|
06486d |
return bt;
|
|
|
06486d |
}
|
|
|
06486d |
|
|
|
06486d |
--
|
|
|
06486d |
1.8.3.1
|
|
|
06486d |
|