diff -r 000000000000 -r 96e5fb8b040d kerneltest/e32utils/d_exc/printstk.pl --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/kerneltest/e32utils/d_exc/printstk.pl Thu Dec 17 09:24:54 2009 +0200 @@ -0,0 +1,584 @@ +#! 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 <>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 = ) + { + 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 () + { + 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 = ) + { + + 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 = ) + { + 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"; + } + +