summaryrefslogtreecommitdiffstats
path: root/lib
diff options
context:
space:
mode:
authorDenys Vlasenko <vda.linux@googlemail.com>2010-02-03 12:55:44 +0100
committerDenys Vlasenko <vda.linux@googlemail.com>2010-02-03 12:55:44 +0100
commit62cb272099cbc2a733a6e0b90c9cdcd8e938a6c7 (patch)
treec35b05684f743815b32138585c0d6c726bf85244 /lib
parent019a1fe365393ac81013ed6984f2eb579de33e85 (diff)
downloadabrt-62cb272099cbc2a733a6e0b90c9cdcd8e938a6c7.tar.gz
abrt-62cb272099cbc2a733a6e0b90c9cdcd8e938a6c7.tar.xz
abrt-62cb272099cbc2a733a6e0b90c9cdcd8e938a6c7.zip
CCpp analyzer: print __glib_assert_msg (rhbz#549735); limit backtrace size
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
Diffstat (limited to 'lib')
-rw-r--r--lib/Plugins/CCpp.cpp43
1 files changed, 35 insertions, 8 deletions
diff --git a/lib/Plugins/CCpp.cpp b/lib/Plugins/CCpp.cpp
index a090a1df..dadff976 100644
--- a/lib/Plugins/CCpp.cpp
+++ b/lib/Plugins/CCpp.cpp
@@ -266,7 +266,7 @@ static void GetBacktrace(const char *pDebugDumpDir,
unsetenv("TERM");
putenv((char*)"TERM=dumb");
- char *args[15];
+ char *args[17];
args[0] = (char*)"gdb";
args[1] = (char*)"-batch";
@@ -294,14 +294,14 @@ static void GetBacktrace(const char *pDebugDumpDir,
*
* Fedora GDB does not strictly need it, it will find the binary
* by its build-id. But for binaries either without build-id
- * (=built on non-Fedora GCC) or which do not have
+ * (= built on non-Fedora GCC) or which do not have
* their debuginfo rpm installed gdb would not find BINARY_FILE
* so it is still makes sense to supply "file BINARY_FILE".
*
* Unfortunately, "file BINARY_FILE" doesn't work well if BINARY_FILE
* was deleted (as often happens during system updates):
* gdb uses specified BINARY_FILE
- * even if it is completely unrelated to the coredump
+ * even if it is completely unrelated to the coredump.
* See https://bugzilla.redhat.com/show_bug.cgi?id=525721
*
* TODO: check mtimes on COREFILE and BINARY_FILE and not supply
@@ -316,16 +316,43 @@ static void GetBacktrace(const char *pDebugDumpDir,
args[7] = (char*)corefile.c_str();
args[8] = (char*)"-ex";
- /* max 3000 frames: with no limit, gdb sometimes OOMs the machine */
- args[9] = (char*)"thread apply all backtrace 3000 full";
+ /*args[9] = ... see below */
args[10] = (char*)"-ex";
args[11] = (char*)"info sharedlib";
/* glibc's abort() stores its message in this variable */
args[12] = (char*)"-ex";
args[13] = (char*)"print (char*)__abort_msg";
- args[14] = NULL;
-
- ExecVP(args, xatoi_u(UID.c_str()), /*redirect_stderr:*/ 1, pBacktrace);
+ args[14] = (char*)"-ex";
+ args[15] = (char*)"print (char*)__glib_assert_msg";
+ args[16] = NULL;
+
+ /* Get the backtrace, but try to cap its size */
+ /* Limit bt depth. With no limit, gdb sometimes OOMs the machine */
+ unsigned bt_depth = 2048;
+ const char *thread_apply_all = "thread apply all ";
+ const char *full = " full";
+ while (1)
+ {
+ string cmd = ssprintf("%sbacktrace %u%s", thread_apply_all, bt_depth, full);
+ args[9] = (char*)cmd.c_str();
+ pBacktrace = "";
+ ExecVP(args, xatoi_u(UID.c_str()), /*redirect_stderr:*/ 1, pBacktrace);
+ if (bt_depth <= 64 || pBacktrace.size() < 256*1024)
+ return;
+ bt_depth /= 2;
+ if (bt_depth <= 64 && thread_apply_all[0] != '\0')
+ {
+ /* This program likely has gazillion threads, dont try to bt them all */
+ bt_depth = 256;
+ thread_apply_all = "";
+ }
+ if (bt_depth <= 64 && full[0] != '\0')
+ {
+ /* Looks like there are gigantic local structures or arrays, disable "full" bt */
+ bt_depth = 256;
+ full = "";
+ }
+ }
}
static void GetIndependentBuildIdPC(const char *unstrip_n_output,