summaryrefslogtreecommitdiff
path: root/libgfortran/runtime
diff options
context:
space:
mode:
authorjb <jb@138bc75d-0d04-0410-961f-82ee72b054a4>2011-05-29 20:13:52 +0000
committerjb <jb@138bc75d-0d04-0410-961f-82ee72b054a4>2011-05-29 20:13:52 +0000
commit45cc458fec7c9f59a7f539d4eff30f05703c8d22 (patch)
treecb0da06274fcd0419a2fe0ae1c347faf09af5f70 /libgfortran/runtime
parent7cb632462af202cfd8fbcf9df675160c17d51fb3 (diff)
downloadgcc-45cc458fec7c9f59a7f539d4eff30f05703c8d22.tar.gz
PR 48931 Use async-signal-safe execve instead of execvp
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@174415 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'libgfortran/runtime')
-rw-r--r--libgfortran/runtime/backtrace.c12
-rw-r--r--libgfortran/runtime/compile_options.c11
-rw-r--r--libgfortran/runtime/main.c39
3 files changed, 60 insertions, 2 deletions
diff --git a/libgfortran/runtime/backtrace.c b/libgfortran/runtime/backtrace.c
index 943332ab615..7d6479fe905 100644
--- a/libgfortran/runtime/backtrace.c
+++ b/libgfortran/runtime/backtrace.c
@@ -104,6 +104,9 @@ fd_gets (char *s, int size, int fd)
}
+extern char *addr2line_path;
+
+
/* show_backtrace displays the backtrace, currently obtained by means of
the glibc backtrace* functions. */
@@ -124,6 +127,9 @@ show_backtrace (void)
#if CAN_PIPE
+ if (addr2line_path == NULL)
+ goto fallback_noerr;
+
/* We attempt to extract file and line information from addr2line. */
do
{
@@ -146,6 +152,7 @@ show_backtrace (void)
/* Child process. */
#define NUM_FIXEDARGS 7
char *arg[NUM_FIXEDARGS];
+ char *newenv[] = { NULL };
close (f[0]);
@@ -160,14 +167,14 @@ show_backtrace (void)
_exit (1);
close (f[1]);
- arg[0] = (char *) "addr2line";
+ arg[0] = addr2line_path;
arg[1] = (char *) "-e";
arg[2] = full_exe_path ();
arg[3] = (char *) "-f";
arg[4] = (char *) "-s";
arg[5] = (char *) "-C";
arg[6] = NULL;
- execvp (arg[0], arg);
+ execve (addr2line_path, arg, newenv);
_exit (1);
#undef NUM_FIXEDARGS
}
@@ -264,6 +271,7 @@ fallback:
#endif /* CAN_PIPE */
+fallback_noerr:
/* Fallback to the glibc backtrace. */
estr_write ("\nBacktrace for this error:\n");
backtrace_symbols_fd (trace, depth, STDERR_FILENO);
diff --git a/libgfortran/runtime/compile_options.c b/libgfortran/runtime/compile_options.c
index dc0da4bcb90..c3e64de33ba 100644
--- a/libgfortran/runtime/compile_options.c
+++ b/libgfortran/runtime/compile_options.c
@@ -58,6 +58,15 @@ backtrace_handler (int signum)
}
+/* Helper function for set_options because we need to access the
+ global variable options which is not seen in set_options. */
+static void
+maybe_find_addr2line (void)
+{
+ if (options.backtrace == -1)
+ find_addr2line ();
+}
+
/* Set the usual compile-time options. */
extern void set_options (int , int []);
export_proto(set_options);
@@ -131,6 +140,8 @@ set_options (int num, int options[])
#if defined(SIGXFSZ)
signal (SIGXFSZ, backtrace_handler);
#endif
+
+ maybe_find_addr2line ();
}
#endif
diff --git a/libgfortran/runtime/main.c b/libgfortran/runtime/main.c
index 54d9e091ce2..bc8dab449e8 100644
--- a/libgfortran/runtime/main.c
+++ b/libgfortran/runtime/main.c
@@ -139,6 +139,40 @@ full_exe_path (void)
}
+char *addr2line_path;
+
+/* Find addr2line and store the path. */
+
+void
+find_addr2line (void)
+{
+#ifdef HAVE_ACCESS
+#define A2L_LEN 10
+ char *path = getenv ("PATH");
+ size_t n = strlen (path);
+ char ap[n + 1 + A2L_LEN];
+ size_t ai = 0;
+ for (size_t i = 0; i < n; i++)
+ {
+ if (path[i] != ':')
+ ap[ai++] = path[i];
+ else
+ {
+ ap[ai++] = '/';
+ memcpy (ap + ai, "addr2line", A2L_LEN);
+ if (access (ap, R_OK|X_OK) == 0)
+ {
+ addr2line_path = strdup (ap);
+ return;
+ }
+ else
+ ai = 0;
+ }
+ }
+#endif
+}
+
+
/* Set the saved values of the command line arguments. */
void
@@ -185,6 +219,9 @@ init (void)
/* if (argc > 1 && strcmp(argv[1], "--resume") == 0) resume(); */
#endif
+ if (options.backtrace == 1)
+ find_addr2line ();
+
random_seed_i4 (NULL, NULL, NULL);
}
@@ -198,4 +235,6 @@ cleanup (void)
if (please_free_exe_path_when_done)
free ((char *) exe_path);
+
+ free (addr2line_path);
}