mirror of
				https://github.com/Rockbox/rockbox.git
				synced 2025-10-24 23:47:38 -04:00 
			
		
		
		
	git-svn-id: svn://svn.rockbox.org/rockbox/trunk@22379 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			272 lines
		
	
	
	
		
			7.6 KiB
		
	
	
	
		
			Perl
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			272 lines
		
	
	
	
		
			7.6 KiB
		
	
	
	
		
			Perl
		
	
	
		
			Executable file
		
	
	
	
	
| #!/usr/bin/env perl
 | |
| #             __________               __   ___.
 | |
| #   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
| #   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
| #   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
| #   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
| #                     \/            \/     \/    \/            \/
 | |
| # $Id$
 | |
| #
 | |
| # Copyright (C) 2009 by Maurus Cuelenaere
 | |
| #
 | |
| # 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 (at your option) any later version.
 | |
| #
 | |
| # This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
| # KIND, either express or implied.
 | |
| #
 | |
| use String::Scanf;
 | |
| use Cwd;
 | |
| 
 | |
| sub check_boundaries
 | |
| {
 | |
|     my $fits = 0, $start = $_[0], $end = $_[0] + $_[1];
 | |
|     foreach my $boundary (@ram_boundaries)
 | |
|     {
 | |
|         if(defined(@$boundary{'name'}) && $start >= @$boundary{'start'} && $end <= @$boundary{'end'})
 | |
|         {
 | |
|             return 1;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| sub dynamic_space
 | |
| {
 | |
|     my $space = $_[0], $space_array = $_[1], $ret;
 | |
| 
 | |
|     printf "This address is in %s space, please select the %s which was used with this address:\n", $space, $space;
 | |
|     $count = 1;
 | |
|     foreach my $el (@$space_array)
 | |
|     {
 | |
|         printf " [%d]: %s\n", $count++, $el;
 | |
|     }
 | |
| 
 | |
|     print "\n";
 | |
|     my $sel = -1;
 | |
|     do
 | |
|     {
 | |
|         print "Selection: ";
 | |
|         $sel = <STDIN>;
 | |
|     } while($sel <= 0 || $sel > $count - 1 || !($sel =~ /^[+-]?\d+$/));
 | |
| 
 | |
|     my $file = sprintf("apps/%ss/%s", $space, @$space_array[$sel - 1]);
 | |
|     $ret{'library'} = sprintf("%s/%s", cwd(), $file);
 | |
|     open FILE, "$objdump -t $file |" or die "Can't open pipe: $!";
 | |
|     while(<FILE>)
 | |
|     {
 | |
|         chomp($_);
 | |
|         if(/^([0-9a-fA-F]+).+\s([0-9a-fA-F]{3,})\s(?:[^\s]+\s)?(.+)$/)
 | |
|         {
 | |
|             (my $addr) = sscanf("%lx", $1);
 | |
|             (my $size) = sscanf("%lx", $2);
 | |
| 
 | |
|             if($lookaddr >= $addr && $lookaddr <= ($addr + $size))
 | |
|             {
 | |
|                 my $diff = abs($lookaddr - $addr);
 | |
|                 if(!defined($ret{'diff'}) || $diff <= $ret{'diff'})
 | |
|                 {
 | |
|                     $ret{'diff'} = $diff;
 | |
|                     $ret{'function'} = $3;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     close FILE;
 | |
| 
 | |
|     return %ret;
 | |
| }
 | |
| 
 | |
| ($lookaddr) = sscanf("0x%lx", $ARGV[0]);
 | |
| ($context_size) = $#ARGV > 0 ? $ARGV[1] : 5;
 | |
| 
 | |
| if($lookaddr != 0)
 | |
| {
 | |
|     # Determine the used objdump utility
 | |
|     open MAKEFILE, "<Makefile" or die "Can't open Makefile: $!";
 | |
|     while(<MAKEFILE>)
 | |
|     {
 | |
|         chomp($_);
 | |
| 
 | |
|         if(/^export OC=(.+)$/)
 | |
|         {
 | |
|             $objdump = $1;
 | |
|             $objdump =~ s/objcopy/objdump/;
 | |
|         }
 | |
|     }
 | |
|     close MAKEFILE;
 | |
| 
 | |
|     # Generate a list of all codecs
 | |
|     open FINDCODECS, "find apps/codecs/ -name '*.elf' 2>&1 |" or die "Can't open pipe: $!";
 | |
|     my @codecs;
 | |
|     while(<FINDCODECS>)
 | |
|     {
 | |
|         chomp($_);
 | |
|         $_ =~ s/apps\/codecs\///;
 | |
|         push(@codecs, $_);
 | |
|     }
 | |
|     close FINDCODECS;
 | |
|     # Generate a list of all plugins
 | |
|     open FINDPLUGINS, "find apps/plugins/ -name '*.elf' 2>&1 |" or die "Can't open pipe: $!";
 | |
|     my @plugins;
 | |
|     while(<FINDPLUGINS>)
 | |
|     {
 | |
|         chomp($_);
 | |
|         $_ =~ s/apps\/plugins\///;
 | |
|         push(@plugins, $_);
 | |
|     }
 | |
|     close FINDPLUGINS;
 | |
| 
 | |
|     open MAPFILE, "<rockbox.map" or die "Can't open rockbox.map: $!";
 | |
|     my $addr, $size, $library, $match, $prev_function, $codec_addr, $plugin_addr;
 | |
|     while(<MAPFILE>)
 | |
|     {
 | |
|         chomp($_);
 | |
| 
 | |
|         if(/^\s*\.text\.([^\s]+)$/)
 | |
|         {
 | |
|             $prev_function = $1;
 | |
|         }
 | |
| 
 | |
|         if(/^\.([^\s]+)\s*(0x[0-9a-fA-F]+)/)
 | |
|         {
 | |
|             ($addr) = sscanf("0x%lx", $2);
 | |
|             if($1 eq "plugin")
 | |
|             {
 | |
|                 $plugin_addr = $addr;
 | |
|             }
 | |
|             elsif($1 eq "codec")
 | |
|             {
 | |
|                 $codec_addr = $addr;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         if(/^.*?\s*(0x[0-9a-fA-F]+)\s*(0x[0-9a-fA-F]+)\s(.+)$/)
 | |
|         {
 | |
|             ($addr)  = sscanf("0x%lx", $1);
 | |
|             ($size)  = sscanf("0x%lx", $2);
 | |
|             $library = $3;
 | |
| 
 | |
|             if(check_boundaries($addr, $size) != 0
 | |
|                && $lookaddr >= $addr && $lookaddr <= ($addr + $size))
 | |
|             {
 | |
|                 #printf "0x%x 0x%x %s %s\n", $addr, $size, $prev_function, $library;
 | |
| 
 | |
|                 my $diff = abs($lookaddr - $addr);
 | |
|                 if(!defined($match{'diff'}) || $diff <= $match{'diff'})
 | |
|                 {
 | |
|                     $match{'diff'} = $diff;
 | |
|                     $match{'library'} = $library;
 | |
|                     $match{'function'} = $prev_function;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         elsif(/^\s*(0x[0-9a-fA-F]+)\s*([^\s]+)$/)
 | |
|         {
 | |
|             ($addr) = sscanf("0x%lx", $1);
 | |
|             my $function = $2;
 | |
| 
 | |
|             if(check_boundaries($addr, 0) != 0 && $lookaddr >= $addr)
 | |
|             {
 | |
|                 #printf "0x%x %s\n", $addr, $function;
 | |
| 
 | |
|                 my $diff = abs($lookaddr - $addr);
 | |
|                 if(!defined($match{'diff'}) || $diff <= $match{'diff'})
 | |
|                 {
 | |
|                     $match{'diff'} = $diff;
 | |
|                     $match{'library'} = $library;
 | |
|                     $match{'function'} = $function;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         elsif(/^(.RAM) *(0x[0-9a-fA-F]+) (0x[0-9a-fA-F]+)/)
 | |
|         {
 | |
|             (my $start_addr) = sscanf("0x%lx", $2);
 | |
|             (my $addr_length) = sscanf("0x%lx", $3);
 | |
|             push(@ram_boundaries, {"name",  $1,
 | |
|                                    "start", $start_addr,
 | |
|                                    "end",   $start_addr + $addr_length
 | |
|                                   });
 | |
|         }
 | |
|     }
 | |
|     close MAPFILE;
 | |
| 
 | |
|     if($lookaddr >= $codec_addr && $lookaddr < $plugin_addr
 | |
|        && $codec_addr != 0)
 | |
|     {
 | |
|         # look for codec
 | |
|         %match = dynamic_space("codec", \@codecs);
 | |
|     }
 | |
|     elsif($lookaddr >= $plugin_addr && $plugin_addr != 0)
 | |
|     {
 | |
|         # look for plugin
 | |
|         %match = dynamic_space("plugin", \@plugins);
 | |
|     }
 | |
| 
 | |
|     printf "%s -> %s\n\n", $match{'library'}, $match{'function'};
 | |
| 
 | |
|     # Replace path/libfoo.a(bar.o) with path/libfoo.a
 | |
|     $match{'library'} =~ s/\(.+\)//;
 | |
| 
 | |
|     open OBJDUMP, "$objdump -S $match{'library'} 2>&1 |" or die "Can't open pipe: $!";
 | |
|     my $found = 0, $addr;
 | |
|     while(<OBJDUMP>)
 | |
|     {
 | |
|         chomp($_);
 | |
| 
 | |
|         if(/^[0-9a-fA-F]+\s\<(.+)\>:$/)
 | |
|         {
 | |
|             $found = ($1 eq $match{'function'});
 | |
|         }
 | |
|         elsif(/Disassembly of section/)
 | |
|         {
 | |
|             $found = 0;
 | |
|         }
 | |
|         elsif($found == 1)
 | |
|         {
 | |
|             if(/^\s*([0-9a-fA-F]+):\s*[0-9a-fA-F]+\s*.+$/)
 | |
|             {
 | |
|                 ($addr) = sscanf("%lx", $1);
 | |
| 
 | |
|                 if($addr - $lookaddr > 0)
 | |
|                 {
 | |
|                     $addr -= $lookaddr;
 | |
|                 }
 | |
|                 if(abs($match{'diff'} - $addr) <= $context_size * 4)
 | |
|                 {
 | |
|                     printf "%s%s\n", ($addr == $match{'diff'} ? ">": " "), $_;
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 # TODO: be able to also show source code (within context_size)
 | |
|                 # printf " %s\n", $_;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     close OBJDUMP;
 | |
| }
 | |
| else
 | |
| {
 | |
|     print "find_addr.pl 0xABCDEF [CONTEXT_SIZE]\n\n";
 | |
|     print <<EOF
 | |
| This makes it possible to find the exact assembly instruction at the specified
 | |
| memory location (depends on Makefile, rockbox.map and the object files).
 | |
| 
 | |
| Usage example:
 | |
|     mcuelenaere\@wim2160:~/rockbox/build2\$ ../utils/analysis/find_addr.pl 0x8001a434 1
 | |
|     /home/mcuelenaere/rockbox/build2/apps/screens.o -> id3_get_info
 | |
| 
 | |
|       23c:  00601021    move    v0,v1
 | |
|     > 240:  80620000    lb  v0,0(v1)
 | |
|       244:  0002180a    movz    v1,zero,v0
 | |
| 
 | |
| 
 | |
| Don't forget to build with -g !
 | |
| EOF
 | |
| ;
 | |
| }
 |