#! perl
# Copyright (c) 2004-2009 Nokia Corporation and/or its subsidiary(-ies).
# All rights reserved.
# This component and the accompanying materials are made available
# under the terms of the License "Eclipse Public License v1.0"
# which accompanies this distribution, and is available
# at the URL "http://www.eclipse.org/legal/epl-v10.html".
#
# Initial Contributors:
# Nokia Corporation - initial contribution.
#
# Contributors:
#
# Description:
#
if (@ARGV<1)
{
#........1.........2.........3.........4.........5.........6.........7.....
print <<USAGE_EOF;
Usage:
printstk.pl d_exc_nnn [romimage.symbol]
Given the output of D_EXC, a file d_exc_nnn.txt and d_exc_nnn.stk, it
uses the other information to try to put symbolic information against
the stack image.
USAGE_EOF
exit 1;
}
sub add_object
{
my ($base, $max, $name) = @_;
$address{$base} = [ $base, $max, $name ];
my $key=$base>>20;
my $maxkey=$max>>20;
while ($key <= $maxkey) # allowing for objects that span the boundary
{
push @{$addresslist{$key}}, $base;
$key+=1;
}
}
my $RomBase = 0xF8000000;
my $RomLimit = 0xFFF00000;
add_object($RomBase,$RomLimit, "ROM");
# Handle a MAKSYM.LOG file for a ROM
#
sub read_rom_symbols
{
my ($romimage)=@_;
open ROMSYMBOLS, $romimage or print "Can't open $romimage\n" and return;
my $a;
my $b;
while (my $line = <ROMSYMBOLS>)
{
if(!($line =~ /^[0-9A-Fa-f]{8}/))
{
next;
}
# 8 bytes for the address
$a = substr $line,0,8;
if(!($a =~ /[0-9A-Fa-f]{8}/))
{
next;
}
# 4 bytes for the length
$b = substr $line,12,4;
if(!($b =~ /[0-9A-Fa-f]{4}/))
{
next;
}
# rest of line is symbol
my $symbol = substr $line,20;
chomp $symbol;
my $base=hex($a);
my $length=hex($b);
if ($base < 0x50000000)
{
next; # skip this line
}
if ($length==0xffffffff)
{
$length=100; # MAKSYM bug? choose a rational length
}
add_object($base, $base+$length-1, $symbol);
}
print "ROM Symbols from $romimage\n";
}
# Handle MAP file for a non execute-in-place binary
#
sub read_map_symbols
{
my ($binary, $binbase)=@_;
$binary =~ /([^\\]+)$/;
my $basename=$1;
if (not open MAPFILE, "$basename.map")
{
print "Can't open map file for \n$binary.map)\n";
return;
}
my @maplines;
while (<MAPFILE>)
{
push @maplines, $_;
}
close MAPFILE;
# See if we're dealing with the RVCT output
if ($maplines[0] =~ /^ARM Linker/)
{
# scroll down to the global symbols
while ($_ = shift @maplines)
{
if (/Global Symbols/)
{
last;
}
}
# .text gets linked at 0x00008000
$imgtext=hex(8000);#start of the text section during linking
foreach (@maplines)
{
# name address ignore size section
if (/^\s*(.+)\s*(0x\S+)\s+[^\d]*(\d+)\s+(.*)$/)
{
my $symbol = $1;
my $addr = hex($2);
my $size = $3;
if ($size > 0)#symbols of the 0 size contain some auxillary information, ignore them
{
add_object($addr-$imgtext+$binbase,#relocated address of the current symbol
$addr-$imgtext+$binbase+$size,#relocated address of the current symbol + size of the current symbol
"$binary $symbol");
}
}
}
}
else
#we are dealing with GCC output
{
my $imgtext;
# Find text section
while (($_ = shift @maplines) && !(/^\.text\s+/))
{
}
/^\.text\s+(\w+)\s+(\w+)/
or die "ERROR: Can't get .text section info for \"$file\"\n";
$imgtext=hex($1);#start of the text section during linking
$binbase-=$imgtext;
foreach (@maplines)
{
if (/___CTOR_LIST__/)
{
last; # end of text section
}
if (/^\s(\.text)?\s+(0x\w+)\s+(0x\w+)\s+(.*)$/io)
{
$textlimit = hex($2)+$binbase+hex($3)-1;
next;
}
if (/^\s+(\w+)\s\s+([a-zA-Z_].+)/o)
{
my $addr = hex($1);
my $symbol = $2;
add_object($addr+$binbase,#relocated address of the current symbol
$textlimit,#limit of the current object section
"$binary $symbol");
next;
}
}
}
#end of GCC output parsing
}
# Handle a matched pair of D_EXC output files (.txt and .stk)
#
sub read_d_exc
{
my ($name)=@_;
$stackbase = 0;
open D_EXC, "$name.txt" or die "Can't open $name.txt\n";
binmode D_EXC;
read D_EXC, $data, 16;
close D_EXC;
if ($data =~ /^(..)*.\0.\0/)
{
# Assuming Unicode
close D_EXC;
# Charconv won't convert STDIN or write to STDOUT
# so we generate an intermediate UTF8 file
system "charconv -little -input=unicode $name.txt -output=utf8 $name.utf8.txt";
open D_EXC, "$name.utf8.txt" or die "Can't open $name.utf8.txt\n";
}
else
{
# Assuming ASCII
open D_EXC, "$name.txt" or die "Can't open $name.txt\n";
}
my $is_eka2_log = 0;
while (my $line = <D_EXC>)
{
if ($line =~ /^EKA2 USER CRASH LOG$/)
{
$is_eka2_log = 1;
next;
}
# code=1 PC=500f7ff8 FAR=00000042 FSR=e8820013
if ($line =~ /^code=\d PC=(.{8})/)
{
$is_exc = 1;
$fault_pc = hex($1);
next;
};
# R13svc=81719fc0 R14svc=50031da0 SPSRsvc=60000010
if ($line =~ /^R13svc=(.{8}) R14svc=(.{8}) SPSRsvc=(.{8})/)
{
$fault_lr = hex($2);
next;
}
# r00=fffffff8 00000000 80000718 80000003
if ($line =~ /^r(\d\d)=(.{8}) (.{8}) (.{8}) (.{8})/)
{
$registers{$1} = $line;
if ($1 == 12)
{
$activesp = hex($3);
$user_pc = hex($5);
$user_lr = hex($4);
}
next;
}
# User Stack 03900000-03905ffb
# EKA1 format deliberately broken (was /^Stack.*/) to catch version problems
if ($line =~ /^User Stack (.{8})-(.{8})/)
{
$stackbase = hex($1);
add_object($stackbase,hex($2), "Stack");
next;
}
# fff00000-fff00fff C:\foo\bar.dll
if ($line =~ /^(.{8})-(.{8}) (.+)/)
{
next if ($RomBase <= hex($1) && hex($1) < $RomLimit); # skip ROM XIP binaries
add_object(hex($1), hex($2), $3);
read_map_symbols($3, hex($1));
}
}
close D_EXC;
die "$name.txt is not a valid EKA2 crash log" unless $is_eka2_log;
if ($stackbase == 0)
{
die "couldn't find stack information in $name.txt\n";
}
die "couldn't find stack pointer in $name.txt\n" unless $activesp != 0;
$activesp -= $stackbase;
# Read in the binary dump of the stack
open STACK, "$name.stk" or die "Can't open $name.stk\n";
print "Stack Data from $name.stk\n";
binmode STACK;
while (read STACK, $data, 4)
{
unshift @stack, (unpack "V", $data);
}
$stackptr = 0;
}
# Handle the captured text output from the Kernel debugger
#
sub read_debugger
{
my ($name)=@_;
open DEBUGFILE, "$name" or die "Can't open $name\n";
print "Kernel Debugger session from $name\n";
# stuff which should be inferred from "$name"
$stackbase = 0x81C00000;
$stackmax = 0x81C01DC0;
$activesp = 0x81c01bc4-$stackbase;
add_object($stackbase,0x81C01FFF, "Stack");
while (my $line = <DEBUGFILE>)
{
if ($line =~ /^(\w{8}): ((\w\w ){16})/)
{
my $addr = hex($1);
if ($addr < $stackbase || $addr > $stackmax)
{
next;
}
if (@stack == 0)
{
if ($addr != $stackbase)
{
printf "Missing stack data for %x-%x - fill with 0x29\n", $stackbase, $addr-1;
@stack = (0x29292929) x (($addr-$stackbase)/4);
}
}
unshift @stack, reverse (unpack "V4", (pack "H2"x16, (split / /,$2)));
}
}
$stackptr = 0;
}
read_d_exc(@ARGV[0]);
if (@ARGV>1)
{
read_rom_symbols(@ARGV[1]);
}
# We've accumulated the ranges of objects indexed by start address,
# with a companion list of addresses subdivided by the leading byte
# Now sort them numerically...
sub numerically { $a <=> $b }
foreach my $key (keys %addresslist)
{
@{$addresslist{$key}} = sort numerically @{$addresslist{$key}};
}
# Off we go, reading the stack!
sub skip_unused
{
my $skipped=0;
while (@stack)
{
my $word=(pop @stack);
if ($word!=0x29292929)
{
push @stack, $word;
last;
}
$skipped += 4;
}
$stackptr += $skipped;
return $skipped;
}
sub lookup_addr
{
my ($word) = @_;
# Optimization - try looking up the address directly
my $base;
my $max;
my $name;
if(defined $address{$word}) {
($base, $max, $name) = @{$address{$word}};
}
if (!(defined $base))
{
my $key=$word>>20;
my $regionbase;
foreach $base (@{$addresslist{$key}})
{
if ($base <= $word)
{
$regionbase = $base;
next;
}
if ($base > $word)
{
last;
}
}
if(defined $regionbase)
{
($base, $max, $name) = @{$address{$regionbase}};
}
}
if (defined $base && defined $max && $base <= $word && $max >= $word)
{
my $data = pack "V", $word;
$data =~ tr [\040-\177]/./c;
return sprintf "%08x %4s %s + 0x%x", $word, $data, $name, $word - $base;
}
return "";
}
sub match_addr
#
# Try matching one of the named areas in the addresslist
#
{
my $word = (pop @stack);
if ($word < 1024*1024)
{
push @stack, $word;
return 0;
}
my $result = lookup_addr($word);
if ($result ne "")
{
print "$result\n";
$stackptr+=4;
return 1;
}
push @stack, $word;
return 0;
}
sub match_tbuf8
#
# Try matching a TBuf8
# 0x3000LLLL 0x0000MMMM data
#
{
if (scalar @stack <3)
{
return 0; # too short
}
my $word = (pop @stack);
my $maxlen = (pop @stack);
my $len = $word & 0x0ffff;
my $type = ($word >> 16) & 0x0ffff;
if ( $type != 0x3000 || $maxlen <= $len || $maxlen > 4* scalar @stack
|| ($stackptr < $activesp && $stackptr + $maxlen + 8 > $activesp))
{
push @stack, $maxlen;
push @stack, $word;
return 0; # wrong type, or invalid looking sizes, or out of date
}
printf "TBuf8<%d>, length %d\n", $maxlen, $len;
$stackptr += 8;
my $string="";
while ($maxlen > 0)
{
$string .= pack "V", pop @stack;
$maxlen -= 4;
$stackptr += 4;
}
if ($len==0)
{
print "\n";
return 1;
}
my $line = substr($string,0,$len);
my @buf = unpack "C*", $line;
$line =~ tr [\040-\177]/./c;
printf "\n %s", $line;
while ($len > 0)
{
my $datalen = 16;
if ($datalen > $len)
{
$datalen = $len;
}
$len -= $datalen;
printf "\n ";
while ($datalen > 0)
{
my $char = shift @buf;
printf "%02x ", $char;
$datalen -= 1;
}
}
printf "\n\n";
return 1;
}
# Skip the unused part of the stack
skip_unused;
printf "High watermark = %04x\n", $stackptr;
# process the interesting bit!
my $printed_current_sp = 0;
while (@stack)
{
if (!$printed_current_sp && $stackptr >= $activesp)
{
printf "\n >>>> current user stack pointer >>>>\n\n";
print $registers{"00"};
print $registers{"04"};
print $registers{"08"};
print $registers{"12"};
if ($is_exc && $user_pc != $fault_pc)
{
print "\nWARNING: A kernel-side exception occured but this script\n";
print "is currently limited to user stack analysis. Sorry.\n";
my $result = lookup_addr($fault_pc);
if ($result ne "")
{
print "Kernel PC = $result\n";
}
$result = lookup_addr($fault_lr);
if ($result ne "")
{
print "Kernel LR = $result\n";
}
print "\n";
}
my $result = lookup_addr($user_pc);
if ($result ne "")
{
print "User PC = $result\n";
}
$result = lookup_addr($user_lr);
if ($result ne "")
{
print "User LR = $result\n";
}
printf "\n >>>> current user stack pointer >>>>\n\n";
$printed_current_sp = 1;
}
printf "%04x ", $stackptr;
match_tbuf8() and next;
match_addr() and next;
$word = pop @stack;
$data = pack "V", $word;
$data =~ tr [\040-\177]/./c;
printf "%08x %4s ", $word, $data;
$stackptr += 4;
if ($word == 0x29292929)
{
$skipped = skip_unused;
if ($skipped != 0)
{
printf "\n....";
}
printf "\n";
next;
}
# Try matching $word against the known addresses of things
printf "\n";
}