imgtools/romtools/maksym/maksymrofs.pl
author Mike Kinghan <mikek@symbian.org>
Mon, 21 Jun 2010 17:57:23 +0100
changeset 589 851206cea67b
parent 0 044383f39525
child 590 360bd6b35136
permissions -rw-r--r--
Add --asm option to elf2e32, so that GCCE builds can use --asm=gas with --dump=a and get a usable .s file (Bug 1405)

#
# Copyright (c) 2005-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: 
# Produces symbolic information given a ROFS log file and .map files for relevant binary files
#

require 5.003_07;
no strict 'vars';
use English;
use FindBin;		# for FindBin::Bin

my $PerlLibPath;    # fully qualified pathname of the directory containing our Perl modules

BEGIN {
    # check user has a version of perl that will cope require 5.005_03;
    # establish the path to the Perl libraries: currently the same directory as this script
    $PerlLibPath = $FindBin::Bin; # X:/epoc32/tools
    $PerlLibPath =~ s/\//\\/g;	# X:\epoc32\tools
    $PerlLibPath .= "\\";
}

use lib $PerlLibPath;
use Modload;
Load_SetModulePath($PerlLibPath);

# Globals
my $maksym = "";
my $rofsbuild;
my $debug = 0;

&args;
&main;
exit 0;

sub CompareAddrs()
{
    return -1 if ($a < $b);
    return 1 if ($a > $b);
    return 0;
}

#
# main
#
sub main()
{
	open (ROFS, "<$rofsbuild") or die "ERROR: Can't open rofsbuild log file \"$rofsbuild\"\n";
	if ($maksym ne "") 
	{
		open (SYM, ">$maksym") or die "ERROR: Can't open output file \"$maksym\"\n";
   		print "Creating $maksym...\n"
	}
	
	my $curretnLine;
	while ($currentLine = <ROFS>) 	
	{
		# Check that the given log file is from a rofs image and set up the name for the symbol file
		if ($currentLine =~ /^Creating Rofs image (\S*)/) 
		{
			if ($maksym eq "") 
			{
				# For backwards compatibility, replace trailing .img with .symbol
				# if no trailing .img, just append .symbol anyway
				$maksym = $1;
				$maksym =~ s/(\.img)?$/.symbol/i;
				close SYM;
				open (SYM, ">$maksym") or die "ERROR: Can't open output file \"$maksym\"\n";
				print "\nCreating $maksym...\n"
			}
			next;
		}
		
		# found at end of file
		if ($currentLine =~ /^Writing Rom image/) 
		{
			close SYM;
			$maksym = "";
			next;
		}
		
		# Data file
		if ($currentLine =~ /^File \'(.*)\' size: \S+\s*$/) 
		{
			my $file = $1;
			$file =~ /([^\\]+)$/;
			printf SYM "\nFrom    $file\n\n00000000    0000    $1\n";
		}

		# Executable file
		elsif ($currentLine =~ /^Compressed executable File \'(.*)\' size: \S+\s*, mode:\S+\s*$/) 
		{
			ProcessCompressedLine($1);
		}
	}
	close SYM;
	close ROFS;
}

sub ProcessCompressedLine
{
	my ($file) = @_;

	my $mapfile;
	my $mapfile2;
	print SYM "\nFrom    $file\n\n";

	# Look in map file for symbols in .text and relocate them
	$mapfile2 = $file.".map";			
	$mapfile = $file;
	$mapfile =~ s/\.\w+$/\.map/;			
	if (!(open (MAP, "$mapfile2") || open (MAP, "$mapfile"))) 
	{
		print "$file\nWarning: Can't open \"$mapfile2\" or \"$mapfile\"\n";
		# couldn't find map file so output in format that is used for non-binary files
		my $BinSize = GetSizeFromBinFile($file);
		$file =~ /([^\\]+)$/;
		printf "00000000    %04x    $1\n", $BinSize;
		printf SYM "00000000    %04x    $1\n", $BinSize;
	} 
	else 
	{
		my @maplines;
		while ($_ = <MAP>) 
		{
			push @maplines, $_;
		}
		close MAP;
		# See if we're dealing with the RVCT output
		if ($file =~m/ARMV5/i) 
		{
			ProcessArmv5File($file, \@maplines);
		} 
		elsif( ($file =~ /GCCE/i) || ($file =~ /ARM4/i) )
		{
			ProcessGcceOrArm4File($file, \@maplines);
		}
		else
		{
			print "\nWarning: cannot determine linker type used to create $file\n";
			$file =~ /([^\\]+)$/;
			printf SYM "00000000    0000    $1\n";
		}
	}
}

sub ProcessArmv5File
{
	my ($file, $mapLines) = @_;
	my @maplines = @$mapLines;
	if ($maplines[0] !~ /^ARM Linker/) 
	{
		print "\nWarning: expecting $file to be generated by ARM linker\n";
		# file not in format produced by ARMV5 linker so treat file as non-binary file
		$file =~ /([^\\]+)$/;
		printf SYM "00000000    0000    $1\n";
	}
	else
	{
		# scroll down to the global symbols
		while ($_ = shift @maplines) 
		{
			if ($_ =~ /Global Symbols/) 
			{
				last;
			}
		}

		my %syms;
		my $baseOffset; # offset to subtract from each address so that the first entry has address 0x0

		foreach (@maplines) 
		{
			# name address ignore size section
			if (/^\s*(.+)\s*(0x\S+)\s+[^\d]*(\d+)\s+(.*)$/) 
			{
				my $sym = $1;
				my $addr = hex($2);
				my $size = sprintf("%04x",$3);
				my $section = $4;
				$size = sprintf("%04x", 8) if ($section =~ /\(StubCode\)/);
				
				# it is possible that there will be more than one entry in a log file for a
				# particular address, this is because aliases are included.
				# The following code checks that the correct function (i.e. the one with
				# non-zero size) is being included in the symbol file.
				if(exists $syms{$addr})
				{ # an entry at this address exists, replace if it is an alias
					if( ($size != 0) && ($addr > 0) )
					{
						if( ! defined $baseOffset )
						{
							$baseOffset = $addr;
						}
						$syms{$addr - $baseOffset} = "$size    $sym $section";
					}
				}
				else
				{ # no entry at this address so create one regardless of whether size is zero
					if( $addr > 0 )
					{
						if( ! defined $baseOffset )
						{
							$baseOffset = $addr;
						}	
						$syms{$addr - $baseOffset} = "$size    $sym $section";
					}
				}

			}
		}

		# Write symbols in address order
		my @addrs = sort CompareAddrs keys %syms;
		for ($i = 0; $i < @addrs ; $i++) 
		{
			my $thisaddr = $addrs[$i];
			printf SYM "%08x    %s\n",
			$thisaddr, $syms{$thisaddr};
		}
	}
}

sub ProcessGcceOrArm4File
{
	my ($file, $mapLines) = @_;
	my @maplines = @$mapLines;
	my %syms;
	my $stubhex=1;

	# 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";

	my $imgtext=hex($1);

	# Slurp symbols 'til the end of the text section
	foreach (@maplines) 
	{

		# blank line marks the end of the text section
		last if (/^$/);

		# .text <addr> <len>  <library(member)>
		# .text$something
		#       <addr> <len>  <library(member)>
		#       <addr> <len>  LONG 0x0

		if (/^\s(\.text)?\s+(0x\w+)\s+(0x\w+)\s+(.*)$/io) 
		{
			my $address = hex($2);
			my $length = hex($3);
			my $libraryfile = $4;
			next if ($libraryfile =~ /^LONG 0x/);
			
			$syms{$address+$length} = ' ';	# impossible symbol as end marker

			#set $stubhex value as $address if there is a match
			if ($libraryfile =~ /.*lib\(.*d\d*s_?\d{5}.o\)$/io) 
			{
				$stubhex=$address;
			}
			next;
		}

		#  <addr>  <symbol name possibly including spaces>
		if (/^\s+(\w+)\s\s+([a-zA-Z_].+)/o) 
		{
			my $addr = hex($1);
			my $symbol = $2;
			$symbol = "stub $symbol" if ($addr == $stubhex);
			$syms{$addr} = $symbol;
			next;
		}
	}				

	# Write symbols in address order
	@addrs = sort CompareAddrs keys %syms;
	for ($i = 0; $i < @addrs - 1; $i++) 
	{
		my $symbol = $syms{$addrs[$i]};
		next if ($symbol eq ' ');
		printf SYM "%08x    %04x    %s\n",
		$addrs[$i] - $imgtext, $addrs[$i+1]-$addrs[$i], $symbol;
	}
	# last address assumed to be imgtext+lentext

	close MAP;
}

#
# args - get command line args
#
sub args
{
	my $arg;
	my @args;
	my $flag;

	&help if (!@ARGV);

	while (@ARGV) 
	{
		$arg = shift @ARGV;

		if ($arg=~/^[\-\/](\S*)$/) 
		{
			$flag=$1;

			if ($flag=~/^[\?h]$/i) 
			{
				&help;
			}
			elsif ($flag=~/^d$/i) 
			{
				$debug = 1;
			}
		       	else 
			{
				print "\nERROR: Unknown flag \"-$flag\"\n";
				&usage;
				exit 1;
			}
		}
		else 
		{
			push @args,$arg;
		}
	}

	if (@args)
	{
		$rofsbuild = shift @args;
		if (@args) 
		{
			$maksym = shift @args;
			if (@args) 
			{
				print "\nERROR: Incorrect argument(s) \"@args\"\n";
				&usage;
				exit 1;
			}
		}
	}
}

sub help ()
{
	my $build;

	&Load_ModuleL('E32TPVER');
	print "\nmaksymrofs - Produce symbolic information given a ROFS image (Build ",
	&E32tpver, ")\n";
	&usage;
	exit 0;
}

sub usage ()
{
    print <<EOF

Usage:
  maksymrofs <logfile> [<outfile>]

Where:
  <logfile>   Log file from rofsbuild tool.
  <outfile>   Output file. Defaults to imagename.symbol.
EOF
    ;
    exit 0;
}

sub GetSizeFromBinFile ()
{
   my ($file) = @_;
   my $tmpfile = "temp.info";
   system("readimage $file -o $tmpfile");
   return 0 if (!-e $tmpfile);
   
   open (TMP, "<$tmpfile");
   my $line;
   my $size = 0;
   while ($line = <TMP>)
   {
      print $line;
      if ($line =~ /^Code size\W+(\w+)$/)
      {
        $size = hex($1);
        last;
      }
   }
   close TMP;
   unlink $tmpfile;
   return $size;
}