summaryrefslogblamecommitdiffstats
path: root/src/util/fnrec.pl
blob: cc7312b61a42ae85206983171e9c2e5a9d4675f6 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16















                                                                     

                                                               































































































































                                                                             
#!/usr/bin/perl -w
#
# Copyright (C) 2010 Michael Brown <mbrown@fensystems.co.uk>.
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License as
# published by the Free Software Foundation; either version 2 of the
# License, or any later version.
#
# This program is distributed in the hope that it will be useful, but
# WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
# General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
# 02110-1301, USA.

=head1 NAME

fnrec.pl

=head1 SYNOPSIS

fnrec.pl [options] bin/image.xxx < logfile

Decode a function trace produced by building with FNREC=1

Options:

	-m,--max-depth=N	Set maximum displayed function depth

=cut

use IPC::Open2;
use Getopt::Long;
use Pod::Usage;
use strict;
use warnings;

use constant MAX_OPEN_BRACE => 10;
use constant MAX_COMMON_BRACE => 3;
use constant MAX_CLOSE_BRACE => 10;

# Parse command-line options
my $max_depth = 16;
Getopt::Long::Configure ( 'bundling', 'auto_abbrev' );
GetOptions (
  'help|h' => sub { pod2usage ( 1 ); },
  'max-depth|m=i' => sub { shift; $max_depth = shift; },
) or die "Could not parse command-line options\n";
pod2usage ( 1 ) unless @ARGV == 1;
my $image = shift;
my $elf = $image.".tmp";
die "ELF file ".$elf." not found\n" unless -e $elf;

# Start up addr2line
my $addr2line_pid = open2 ( my $addr2line_out, my $addr2line_in,
			    "addr2line", "-f", "-e", $elf )
    or die "Could not start addr2line: $!\n";

# Translate address using addr2line
sub addr2line {
  my $address = shift;

  print $addr2line_in $address."\n";
  chomp ( my $name = <$addr2line_out> );
  chomp ( my $file_line = <$addr2line_out> );
  ( my $file, my $line ) = ( $file_line =~ /^(.*):(\d+)$/ );
  $file =~ s/^.*\/src\///;
  my $location = ( $line ? $file.":".$line." = ".$address : $address );
  return ( $name, $location );
}

# Parse logfile
my $depth = 0;
my $depths = [];
while ( my $line = <> ) {
  chomp $line;
  $line =~ s/\r//g;
  ( my $called_fn, my $call_site, my $entry_count, my $exit_count ) =
      ( $line =~ /^(0x[0-9a-f]+)\s+(0x[0-9a-f]+)\s+([0-9]+)\s+([0-9]+)$/ )
      or print $line."\n" and next;

  ( my $called_fn_name, undef ) = addr2line ( $called_fn );
  ( undef, my $call_site_location ) = addr2line ( $call_site );
  $entry_count = ( $entry_count + 0 );
  $exit_count = ( $exit_count + 0 );

  if ( $entry_count >= $exit_count ) {
    #
    # Function entry
    #
    my $text = "";
    $text .= $called_fn_name." (from ".$call_site_location.")";
    if ( $exit_count <= MAX_COMMON_BRACE ) {
      $text .= " { }" x $exit_count;
    } else {
      $text .= " { } x ".$exit_count;
    }
    $entry_count -= $exit_count;
    if ( $entry_count <= MAX_OPEN_BRACE ) {
      $text .= " {" x $entry_count;
    } else {
      $text .= " { x ".$entry_count;
    }
    my $indent = "  " x $depth;
    print $indent.$text."\n";
    $depth += $entry_count;
    $depth = $max_depth if ( $depth > $max_depth );
    push @$depths, ( { called_fn => $called_fn, call_site => $call_site } ) x
	( $depth - @$depths );
  } else {
    #
    # Function exit
    #
    my $text = "";
    if ( $entry_count <= MAX_COMMON_BRACE ) {
      $text .= " { }" x $entry_count;
    } else {
      $text .= " { } x ".$entry_count;
    }
    $exit_count -= $entry_count;
    if ( $exit_count <= MAX_CLOSE_BRACE ) {
      $text .= " }" x $exit_count;
    } else {
      $text .= " } x ".$exit_count;
    }
    $depth -= $exit_count;
    $depth = 0 if ( $depth < 0 );
    if ( ( @$depths == 0 ) ||
	 ( $depths->[$depth]->{called_fn} ne $called_fn ) ||
	 ( $depths->[$depth]->{call_site} ne $call_site ) ) {
      $text .= " (from ".$called_fn_name." to ".$call_site_location.")";
    }
    splice ( @$depths, $depth );
    my $indent = "  " x $depth;
    print substr ( $indent.$text, 1 )."\n";
  }
}

# Clean up addr2line
close $addr2line_in;
close $addr2line_out;
waitpid ( $addr2line_pid, 0 );