Printing back trace of functions inside a program
Suppose you have a software that gets segmentation faults, the default behavior is to terminate the application, however if your application runs on a remote system, you need a way to find out why and where you got the segmentation fault.
One way is of course to generate a core dump and later use gdb with the bt command to print the stack back trace, however a core dump is a large file consuming disk space and if you have several crashes of the application you need a way to log errors.
One such a way is to use the backtrace() function in your program to print the last functions addresses.
Of course function addresses are not exactly what you want but you can get the function names by searching the map file generated by your linker (option -Map filename to linker).
You can trap the SIGSEGV and call a function that will print the backtrace.
The following code shows a function to print the last 10 functions on the stack:
#include
#include /* for backtrace */
void bt(void) {
int c, i;
void *addresses[10];
c = backtrace(addresses, 10);
printf("backtrace returned: %dn", c);
for(i = 0; i < c; i++) {
printf("%d: %Xn", i, (int)addresses[i]);
}
}
Note: After this tip was published I got the following tip from Rami Rosen:
You can use the backtrace_symbols if you use the -rdynamic switch to gcc so the program will look as follows:
#include
#include /* for backtrace */
void bt(void) {
int c, i;
void *addresses[10];
char **strings;
c = backtrace(addresses, 10);
strings = backtrace_symbols(addresses,c);
printf("backtrace returned: %dn", c);
for(i = 0; i < c; i++) {
printf("%d: %X ", i, (int)addresses[i]);
printf("%sn", strings[i]); }
}
}
Thanks again to Rami Rosen for the tip.
This of course prints to the standard out, but it would be easy to change it to print to a log file.
You can call this function from a signal handler.
To set a signal handler use the signal() function.
function RecursiveDelete($num) {
global $commentstbl;
$query = "SELECT num FROM $commentstbl WHERE ancestor='$num'";
$result = DoQuery($query, __LINE__);
if($result->num_rows == 0)
return; /* end of recursive call */
while($line = $result->fetch_assoc()) {
$n = $line['num'];
$query = "DELETE FROM comments WHERE num=$n";
$newresult = DoQuery($query, __LINE__);
RecursiveDelete($n);
}
}
function GetTimeStr($hour, $min, $sec, $month, $day, $year) {
global $timeoffset;
$time = mktime($hour, $min, $sec, $month, $day, $year, 0);
$time += ($timeoffset * 3600); /* add time offset in seconds */
$str = date("d/m/Y H:i:s", $time);
return $str;
}
/*
| AddLinks:
| Replace URL's in string with links
*/
function AddLinks($string) {
$string = preg_replace("/(^|[^=\"\/])\b((\w+:\/\/|www\.)[^\s<]+)".
"((\W+|\b)([\s<]|$))/i", "$1
$2$4",
$string);
return preg_replace("/href=\"www/i", "href=\"http://www", $string);
// $txt = preg_replace( "/(?\s]+)/i", "
\\0", $txt );
// "
}
/*
| Replace special codes in messages
*/
function SpecialCodes($str) {
$str = stripslashes($str);
$str = str_replace('<', '<', $str);
$str = str_replace('|code|', '
', $str);
$str = str_replace('|CODE|', '', $str);
$str = str_replace('|ECODE|', '
', $str);
$str = str_replace('|ecode|', '
', $str);
$str = str_replace('|b|', '
', $str);
$str = str_replace('|B|', '', $str);
$str = str_replace('|eb|', '', $str);
$str = str_replace('|EB|', '', $str);
$str = str_replace('|u|', '
', $str);
$str = str_replace('|U|', '', $str);
$str = str_replace('|eu|', '', $str);
$str = str_replace('|EU|', '', $str);
$str = str_replace("\n", "
", $str);
$str = AddLinks($str);
return $str;
}
/*
| DisplayContents:
| Show a string of section body, parsing dot commands
| currently only one command is supported
| .C that will start or stop code section
*/
function DisplayContents($contents) {
$incode = 0;
$contents = SpecialCodes($contents);
$a = explode('
', $contents);
foreach($a as $val) {
if($val[0] == '.') { /* this is a command */
if($val[1] == 'C') {
if(!$incode) {
print "
\n";
$incode++;
}
else {
print "
\n";
$incode--;
}
}
}
else {
$val = AddLinks($val);
print "$val
\n";
}
}
if($incode) {
while($incode) {
print "
\n";
$incode--;
}
}
}
/*
| DispComments:
| Recursive function to display comments for specific article
*/
function DispComments($id, $ancestor, $level, $delprivilege) {
global $commentstbl, $abspath, $l10n;
global $dir, $lang;
global $base;
// print "dir: $dir
\n";
$query = "SELECT * FROM $commentstbl WHERE article='$id' AND ancestor='$ancestor' ORDER BY time DESC";
// print "Query: $query
\n";
$result = DoQuery($query, __LINE__);
if($result->num_rows == 0)
return; /* end of recursive call */
if($level == 0)
print "
\n"; /* spacer between two threades */
while($line = $result->fetch_assoc()) {
$msgnum = $line['num'];
$title = SpecialCodes($line['title']);
if(empty($title))
$title = _("No title");
$msguser = $line['name'];
$email = $line['email'];
$website = $line['website'];
$comment = $line['comment'];
$time = $line['time'];
$i = $level * 10;
if($dir == 'rtl')
$style = "margin-right:${i}px";
else
$style = "margin-left:${i}px";
print "
\n";
DispComments($id, $msgnum, $level+1, $delprivilege, $dir);
}
}
if($action == 'delcomment') {
$num = $_GET['num'];
$query = "DELETE FROM $commentstbl WHERE num='$num'";
$result = DoQuery($query, __LINE__);
RecursiveDelete($num);
$url = urldecode($id);
print "
\n";
return;
}
$commenturl1 = urlencode($commenturl);
$url = "${base}articlemsg.php?id=$commenturl1&ancestor=0&lang=$lang";
$url = urlencode($url);
print "