kerneltest/e32utils/d_exc/printstk.pl
changeset 9 96e5fb8b040d
equal deleted inserted replaced
-1:000000000000 9:96e5fb8b040d
       
     1 #! perl
       
     2 # Copyright (c) 2004-2009 Nokia Corporation and/or its subsidiary(-ies).
       
     3 # All rights reserved.
       
     4 # This component and the accompanying materials are made available
       
     5 # under the terms of the License "Eclipse Public License v1.0"
       
     6 # which accompanies this distribution, and is available
       
     7 # at the URL "http://www.eclipse.org/legal/epl-v10.html".
       
     8 #
       
     9 # Initial Contributors:
       
    10 # Nokia Corporation - initial contribution.
       
    11 #
       
    12 # Contributors:
       
    13 #
       
    14 # Description:
       
    15 #
       
    16 
       
    17 if (@ARGV<1)
       
    18 	{
       
    19 #........1.........2.........3.........4.........5.........6.........7.....
       
    20 	print <<USAGE_EOF;
       
    21 
       
    22 Usage:
       
    23 	printstk.pl d_exc_nnn [romimage.symbol]
       
    24 
       
    25 Given the output of D_EXC, a file d_exc_nnn.txt and d_exc_nnn.stk, it 
       
    26 uses the other information to try to put symbolic information against 
       
    27 the stack image.
       
    28 
       
    29 USAGE_EOF
       
    30 	exit 1;
       
    31 	}
       
    32 
       
    33 sub add_object
       
    34 	{
       
    35 	my ($base, $max, $name) = @_;
       
    36 	$address{$base} = [ $base, $max, $name ];
       
    37 	my $key=$base>>20;
       
    38 	my $maxkey=$max>>20;
       
    39 	while ($key <= $maxkey)		# allowing for objects that span the boundary
       
    40 		{
       
    41 		push @{$addresslist{$key}}, $base;
       
    42 		$key+=1;
       
    43 		}
       
    44 	}
       
    45 
       
    46 my $RomBase = 0xF8000000;
       
    47 my $RomLimit = 0xFFF00000;
       
    48 add_object($RomBase,$RomLimit, "ROM");
       
    49 
       
    50 # Handle a MAKSYM.LOG file for a ROM
       
    51 #
       
    52 sub read_rom_symbols
       
    53 	{
       
    54 	my ($romimage)=@_;
       
    55 	open ROMSYMBOLS, $romimage or print "Can't open $romimage\n" and return;
       
    56 
       
    57 	my $a;
       
    58 	my $b;
       
    59 	while (my $line = <ROMSYMBOLS>)
       
    60 		{
       
    61 		if(!($line =~ /^[0-9A-Fa-f]{8}/))
       
    62 			{
       
    63 			next;
       
    64 			}
       
    65 		# 8 bytes for the address
       
    66 		
       
    67 		$a = substr $line,0,8;
       
    68 		if(!($a =~ /[0-9A-Fa-f]{8}/))
       
    69 			{
       
    70 			next;
       
    71 			}
       
    72 		# 4 bytes for the length
       
    73 		$b = substr $line,12,4;
       
    74 		if(!($b =~ /[0-9A-Fa-f]{4}/))
       
    75 			{
       
    76 			next;
       
    77 			}
       
    78 		# rest of line is symbol
       
    79 		my $symbol = substr $line,20;
       
    80 		chomp $symbol;
       
    81 
       
    82 		my $base=hex($a);
       
    83 		my $length=hex($b);
       
    84 		if ($base < 0x50000000) 
       
    85 			{
       
    86 			next;	# skip this line
       
    87 			}
       
    88 		if ($length==0xffffffff)
       
    89 			{
       
    90 			$length=100;	# MAKSYM bug? choose a rational length
       
    91 			}
       
    92 		add_object($base, $base+$length-1, $symbol);
       
    93 		}
       
    94 	print "ROM Symbols from $romimage\n";
       
    95 	}
       
    96 
       
    97 # Handle MAP file for a non execute-in-place binary
       
    98 #
       
    99 sub read_map_symbols
       
   100 	{
       
   101 	my ($binary, $binbase)=@_;
       
   102 	$binary =~ /([^\\]+)$/;
       
   103 	my $basename=$1;
       
   104 	if (not open MAPFILE, "$basename.map")
       
   105 		{
       
   106 		print "Can't open map file for \n$binary.map)\n";		
       
   107 		return;
       
   108 		}
       
   109 
       
   110 		
       
   111 	my @maplines;
       
   112 	while (<MAPFILE>) 
       
   113 		{
       
   114 		push @maplines, $_;
       
   115 		}
       
   116 	close MAPFILE;
       
   117 # See if we're dealing with the RVCT output
       
   118 	if ($maplines[0] =~ /^ARM Linker/) 
       
   119 		{
       
   120 		# scroll down to the global symbols
       
   121 		while ($_ = shift @maplines) 
       
   122 			{
       
   123 			if (/Global Symbols/) 
       
   124 				{
       
   125 				last;
       
   126 				}
       
   127 			}
       
   128 		# .text gets linked at 0x00008000		
       
   129 		$imgtext=hex(8000);#start of the text section during linking
       
   130 		
       
   131 		foreach (@maplines) 
       
   132 			{
       
   133 			# name address ignore size section
       
   134 			if (/^\s*(.+)\s*(0x\S+)\s+[^\d]*(\d+)\s+(.*)$/) 
       
   135 				{
       
   136 				my $symbol  = $1;
       
   137 				my $addr = hex($2);
       
   138 				my $size = $3;
       
   139 				if ($size > 0)#symbols of the 0 size contain some auxillary information, ignore them
       
   140 					{
       
   141 	            			add_object($addr-$imgtext+$binbase,#relocated address of the current symbol 
       
   142 						$addr-$imgtext+$binbase+$size,#relocated address of the current symbol + size of the current symbol
       
   143 						"$binary $symbol");
       
   144 					}
       
   145 				}
       
   146 			}			      
       
   147 		}
       
   148 	else 
       
   149 #we are dealing with GCC output
       
   150 		{
       
   151 		my $imgtext;
       
   152 		
       
   153 		# Find text section
       
   154 		while (($_ = shift @maplines) && !(/^\.text\s+/)) 
       
   155 			{
       
   156 			}
       
   157 
       
   158 		/^\.text\s+(\w+)\s+(\w+)/
       
   159 			or die "ERROR: Can't get .text section info for \"$file\"\n";
       
   160 		$imgtext=hex($1);#start of the text section during linking
       
   161 		$binbase-=$imgtext;
       
   162 		
       
   163 		foreach (@maplines) 
       
   164 			{
       
   165 			if (/___CTOR_LIST__/)
       
   166 				{
       
   167 				last;	# end of text section
       
   168 				}
       
   169 
       
   170 			if (/^\s(\.text)?\s+(0x\w+)\s+(0x\w+)\s+(.*)$/io) 
       
   171 				{		    
       
   172 				$textlimit = hex($2)+$binbase+hex($3)-1;			
       
   173 				next;
       
   174 				}
       
   175 					
       
   176 			if (/^\s+(\w+)\s\s+([a-zA-Z_].+)/o) 			
       
   177 				{				    
       
   178 				my $addr = hex($1);
       
   179 				my $symbol = $2;
       
   180 				add_object($addr+$binbase,#relocated address of the current symbol
       
   181 					$textlimit,#limit of the current object section
       
   182 					"$binary $symbol");
       
   183 				next;
       
   184 				}
       
   185 			}						
       
   186 		}
       
   187 #end of GCC output parsing		
       
   188 	}
       
   189 
       
   190 # Handle a matched pair of D_EXC output files (.txt and .stk)
       
   191 #
       
   192 sub read_d_exc
       
   193 	{
       
   194 	my ($name)=@_;
       
   195 
       
   196 	$stackbase = 0;
       
   197 	open D_EXC, "$name.txt" or die "Can't open $name.txt\n";
       
   198 
       
   199 	binmode D_EXC;
       
   200 	read D_EXC, $data, 16;
       
   201 	close D_EXC;
       
   202 
       
   203 	if ($data =~ /^(..)*.\0.\0/)
       
   204 		{
       
   205 		# Assuming Unicode
       
   206 		close D_EXC;
       
   207 
       
   208 		# Charconv won't convert STDIN or write to STDOUT
       
   209 		# so we generate an intermediate UTF8 file 
       
   210 		system "charconv -little -input=unicode $name.txt -output=utf8 $name.utf8.txt";
       
   211 
       
   212 		open D_EXC, "$name.utf8.txt" or die "Can't open $name.utf8.txt\n";
       
   213 		}
       
   214 	else
       
   215 		{
       
   216 		# Assuming ASCII
       
   217 		open D_EXC, "$name.txt" or die "Can't open $name.txt\n";
       
   218 		}
       
   219 
       
   220 	my $is_eka2_log = 0;
       
   221 
       
   222 	while (my $line = <D_EXC>)
       
   223 		{
       
   224 
       
   225 		if ($line =~ /^EKA2 USER CRASH LOG$/)
       
   226 			{
       
   227 			$is_eka2_log = 1;
       
   228 			next;
       
   229 			}
       
   230 	
       
   231 		# code=1 PC=500f7ff8 FAR=00000042 FSR=e8820013
       
   232 		
       
   233 		if ($line =~ /^code=\d PC=(.{8})/)
       
   234 			{
       
   235 			$is_exc = 1;
       
   236 			$fault_pc = hex($1);
       
   237 			next;
       
   238 			};
       
   239 
       
   240 		# R13svc=81719fc0 R14svc=50031da0 SPSRsvc=60000010
       
   241 	
       
   242 		if ($line =~ /^R13svc=(.{8}) R14svc=(.{8}) SPSRsvc=(.{8})/)
       
   243 			{
       
   244 			$fault_lr = hex($2);
       
   245 			next;
       
   246 			}
       
   247 
       
   248 		# r00=fffffff8 00000000 80000718 80000003
       
   249 
       
   250 		if ($line =~ /^r(\d\d)=(.{8}) (.{8}) (.{8}) (.{8})/)
       
   251 			{
       
   252 			$registers{$1} = $line;
       
   253 			if ($1 == 12)
       
   254 				{
       
   255 				$activesp = hex($3);
       
   256 				$user_pc = hex($5);
       
   257 				$user_lr = hex($4);
       
   258 				}
       
   259 			next;
       
   260 			}
       
   261 
       
   262 		# User Stack 03900000-03905ffb
       
   263 		# EKA1 format deliberately broken (was /^Stack.*/) to catch version problems
       
   264 
       
   265 		if ($line =~ /^User Stack (.{8})-(.{8})/)
       
   266 			{
       
   267 			$stackbase = hex($1);
       
   268 			add_object($stackbase,hex($2), "Stack");
       
   269 			next;
       
   270 			}
       
   271 
       
   272 		# fff00000-fff00fff C:\foo\bar.dll
       
   273 
       
   274 		if ($line =~ /^(.{8})-(.{8}) (.+)/)
       
   275 			{
       
   276 			next if ($RomBase <= hex($1) && hex($1) < $RomLimit); # skip ROM XIP binaries
       
   277 			add_object(hex($1), hex($2), $3);
       
   278 			read_map_symbols($3, hex($1));
       
   279 			}
       
   280 		}
       
   281 	close D_EXC;
       
   282 
       
   283 	die "$name.txt is not a valid EKA2 crash log" unless $is_eka2_log;
       
   284 
       
   285 	if ($stackbase == 0)
       
   286 		{
       
   287 		die "couldn't find stack information in $name.txt\n";
       
   288 		}
       
   289 
       
   290 	die "couldn't find stack pointer in  $name.txt\n" unless $activesp != 0;
       
   291 	$activesp -= $stackbase;
       
   292 
       
   293 	# Read in the binary dump of the stack
       
   294 
       
   295 	open STACK, "$name.stk" or die "Can't open $name.stk\n";
       
   296 	print "Stack Data from $name.stk\n";
       
   297 
       
   298 	binmode STACK;
       
   299 	while (read STACK, $data, 4)
       
   300 		{
       
   301 		unshift @stack, (unpack "V", $data);
       
   302 		}
       
   303 	$stackptr = 0;
       
   304 	}
       
   305 
       
   306 # Handle the captured text output from the Kernel debugger
       
   307 #
       
   308 sub read_debugger
       
   309 	{
       
   310 	my ($name)=@_;
       
   311 
       
   312 	open DEBUGFILE, "$name" or die "Can't open $name\n";
       
   313 	print "Kernel Debugger session from $name\n";
       
   314 
       
   315 	# stuff which should be inferred from "$name"
       
   316 
       
   317 	$stackbase = 0x81C00000;
       
   318 	$stackmax  = 0x81C01DC0;
       
   319 	$activesp = 0x81c01bc4-$stackbase;
       
   320 	add_object($stackbase,0x81C01FFF, "Stack");
       
   321 
       
   322 	while (my $line = <DEBUGFILE>)
       
   323 		{
       
   324 		if ($line =~ /^(\w{8}): ((\w\w ){16})/)
       
   325 			{
       
   326 			my $addr = hex($1);
       
   327 			if ($addr < $stackbase || $addr > $stackmax)
       
   328 				{
       
   329 				next;
       
   330 				}
       
   331 			if (@stack == 0)
       
   332 				{
       
   333 				if ($addr != $stackbase)
       
   334 					{
       
   335 					printf "Missing stack data for %x-%x - fill with 0x29\n", $stackbase, $addr-1;
       
   336 					@stack = (0x29292929) x (($addr-$stackbase)/4);
       
   337 					}
       
   338 				}
       
   339 			unshift @stack, reverse (unpack "V4", (pack "H2"x16, (split / /,$2)));
       
   340 			}
       
   341 		}
       
   342 		$stackptr = 0;
       
   343 	}
       
   344 
       
   345 read_d_exc(@ARGV[0]);
       
   346 if (@ARGV>1)
       
   347 	{
       
   348 	read_rom_symbols(@ARGV[1]);
       
   349 	}
       
   350 
       
   351 # We've accumulated the ranges of objects indexed by start address,
       
   352 # with a companion list of addresses subdivided by the leading byte
       
   353 # Now sort them numerically...
       
   354 
       
   355 sub numerically { $a <=> $b }
       
   356 foreach my $key (keys %addresslist)
       
   357 	{
       
   358 	@{$addresslist{$key}} = sort numerically @{$addresslist{$key}};
       
   359 	}
       
   360 
       
   361 # Off we go, reading the stack!
       
   362 
       
   363 sub skip_unused 
       
   364 	{
       
   365 	my $skipped=0;
       
   366 	while (@stack)
       
   367 		{
       
   368 		my $word=(pop @stack);
       
   369 		if ($word!=0x29292929)
       
   370 			{ 
       
   371 			push @stack, $word;
       
   372 			last;
       
   373 			}
       
   374 		$skipped += 4;
       
   375 		}
       
   376 	$stackptr += $skipped;
       
   377 	return $skipped;
       
   378 	}
       
   379 
       
   380 sub lookup_addr
       
   381 {
       
   382 	my ($word) = @_;
       
   383 
       
   384 	# Optimization - try looking up the address directly
       
   385 
       
   386 	my $base;
       
   387 	my $max;
       
   388 	my $name;
       
   389 	if(defined $address{$word}) {
       
   390 		($base, $max, $name) = @{$address{$word}};
       
   391 	}
       
   392 	if (!(defined $base))
       
   393 		{
       
   394 		my $key=$word>>20;
       
   395 		my $regionbase;
       
   396 		foreach $base (@{$addresslist{$key}})
       
   397 			{
       
   398 			if ($base <= $word)
       
   399 				{
       
   400 				$regionbase = $base;
       
   401 				next;
       
   402 				}
       
   403 			if ($base > $word)
       
   404 				{
       
   405 				last;
       
   406 				}
       
   407 			}
       
   408 		if(defined $regionbase)
       
   409 			{
       
   410 			($base, $max, $name) = @{$address{$regionbase}};
       
   411 			}
       
   412 		}
       
   413 	if (defined $base && defined $max && $base <= $word && $max >= $word)
       
   414 		{
       
   415 		my $data = pack "V", $word;
       
   416 		$data =~ tr [\040-\177]/./c;
       
   417 		return sprintf "%08x %4s  %s + 0x%x", $word, $data, $name, $word - $base;
       
   418 		}
       
   419 	return "";
       
   420 }
       
   421 
       
   422 sub match_addr
       
   423 #
       
   424 # Try matching one of the named areas in the addresslist
       
   425 #
       
   426 {
       
   427 	my $word = (pop @stack);
       
   428 
       
   429 	if ($word < 1024*1024)
       
   430 		{
       
   431 		push @stack, $word;
       
   432 		return 0;
       
   433 		}
       
   434 
       
   435 	my $result = lookup_addr($word);
       
   436 	if ($result ne "")
       
   437 		{
       
   438 		print "$result\n";
       
   439 		$stackptr+=4;
       
   440 		return 1;
       
   441 		}
       
   442 	push @stack, $word;
       
   443 	return 0;
       
   444 	}
       
   445 
       
   446 sub match_tbuf8
       
   447 #
       
   448 # Try matching a TBuf8
       
   449 #	0x3000LLLL 0x0000MMMM data
       
   450 #	
       
   451 	{
       
   452 	if (scalar @stack <3)
       
   453 		{
       
   454 		return 0;	# too short
       
   455 		}
       
   456 	my $word = (pop @stack);
       
   457 	my $maxlen = (pop @stack);
       
   458 	
       
   459 	my $len = $word & 0x0ffff;
       
   460 	my $type = ($word >> 16) & 0x0ffff;
       
   461 	if ( $type != 0x3000 || $maxlen <= $len || $maxlen > 4* scalar @stack 
       
   462 		|| ($stackptr < $activesp && $stackptr + $maxlen + 8 > $activesp))
       
   463 		{
       
   464 		push @stack, $maxlen;
       
   465 		push @stack, $word;
       
   466 		return 0;		# wrong type, or invalid looking sizes, or out of date
       
   467 		}
       
   468 
       
   469 	printf "TBuf8<%d>, length %d\n", $maxlen, $len;
       
   470 	$stackptr += 8;
       
   471 
       
   472 	my $string="";
       
   473 	while ($maxlen > 0)
       
   474 		{
       
   475 		$string .= pack "V", pop @stack;
       
   476 		$maxlen -= 4;
       
   477 		$stackptr += 4;
       
   478 		}
       
   479 	if ($len==0)
       
   480 		{
       
   481 		print "\n";
       
   482 		return 1;
       
   483 		}
       
   484 	my $line = substr($string,0,$len);
       
   485 	my @buf = unpack "C*", $line;
       
   486 	$line =~ tr [\040-\177]/./c;
       
   487 	printf "\n  %s", $line;
       
   488 	while ($len > 0)
       
   489 		{
       
   490 		my $datalen = 16;
       
   491 		if ($datalen > $len)
       
   492 			{
       
   493 			$datalen = $len;
       
   494 			}
       
   495 		$len -= $datalen;
       
   496 		printf "\n  ";
       
   497 		while ($datalen > 0)
       
   498 			{
       
   499 			my $char = shift @buf;
       
   500 			printf "%02x ", $char;
       
   501 			$datalen -= 1;
       
   502 			}
       
   503 		}
       
   504 	printf "\n\n";
       
   505 	return 1;
       
   506 	}
       
   507 
       
   508 # Skip the unused part of the stack
       
   509 
       
   510 skip_unused;
       
   511 printf "High watermark = %04x\n", $stackptr;
       
   512 
       
   513 # process the interesting bit!
       
   514 
       
   515 my $printed_current_sp = 0;
       
   516 while (@stack)
       
   517 	{
       
   518 	if (!$printed_current_sp && $stackptr >= $activesp)
       
   519 		{
       
   520 		printf "\n >>>> current user stack pointer >>>>\n\n";
       
   521 
       
   522 		print $registers{"00"};
       
   523 		print $registers{"04"};
       
   524 		print $registers{"08"};
       
   525 		print $registers{"12"};
       
   526 
       
   527 		if ($is_exc && $user_pc != $fault_pc)
       
   528 			{
       
   529 			print "\nWARNING: A kernel-side exception occured but this script\n";
       
   530 			print "is currently limited to user stack analysis. Sorry.\n";
       
   531 			my $result = lookup_addr($fault_pc);
       
   532 			if ($result ne "")
       
   533 				{
       
   534 				print "Kernel PC = $result\n";
       
   535 				}
       
   536 			$result = lookup_addr($fault_lr);
       
   537 			if ($result ne "")
       
   538 				{
       
   539 				print "Kernel LR = $result\n";
       
   540 				}
       
   541 			print "\n";
       
   542 			}
       
   543 
       
   544 		my $result = lookup_addr($user_pc);
       
   545 		if ($result ne "")
       
   546 			{
       
   547 			print "User PC = $result\n";
       
   548 			}
       
   549 		$result = lookup_addr($user_lr);
       
   550 		if ($result ne "")
       
   551 			{
       
   552 			print "User LR = $result\n";
       
   553 			}
       
   554 		printf "\n >>>> current user stack pointer >>>>\n\n";
       
   555 		$printed_current_sp = 1;
       
   556 		}
       
   557 
       
   558 	printf "%04x  ", $stackptr;
       
   559 
       
   560 	match_tbuf8() and next;
       
   561 	match_addr() and next;
       
   562 
       
   563 	$word = pop @stack;
       
   564 	$data = pack "V", $word;
       
   565 	$data =~ tr [\040-\177]/./c;
       
   566 	printf "%08x %4s  ", $word, $data;
       
   567 	$stackptr += 4;
       
   568 
       
   569 	if ($word == 0x29292929)
       
   570 		{
       
   571 		$skipped = skip_unused;
       
   572 		if ($skipped != 0)
       
   573 			{
       
   574 			printf "\n....";
       
   575 			}
       
   576 		printf "\n";
       
   577 		next;
       
   578 		}
       
   579 
       
   580 	# Try matching $word against the known addresses of things
       
   581 	printf "\n";
       
   582 	}
       
   583 
       
   584