kerneltest/e32utils/trace/btracevw.pl
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Mon, 18 Jan 2010 21:31:10 +0200
changeset 11 329ab0095843
parent 9 96e5fb8b040d
permissions -rw-r--r--
Revision: 201003 Kit: 201003

#
# Copyright (c) 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:
#

#!/usr/bin/perl

use File::Find;
use File::Spec::Functions;


	my $TraceFileName;
	
	my $PrintFlagFilePos = 0;
	my $PrintFlagHdrLen = 0;
	my $PrintFlagHdrFlags = 0;
	my $PrintFlagFormatString = 0;
	my $VerboseMode = 0;
	my $RawMode = 0;
	my $FormatIdIsSubCategory = 0;
	my $OutputSawDictionaryMode = 0;
	
	# for the category range 0-191, the format string is indexed by the category & subcategory
	%FormatTables = 
		(
		0 => 			# ERDebugPrintf
			{
			0 => "ThreadId %h, %s",
			},
	
		1 => 			# ERKernPrintf
			{
			0 => "ThreadId %h, %s",
			},

		3 =>			# EThreadIdentification
			{	
			0 => "ENanoThreadCreate, NThread %x",
			1 => "ENanoThreadDestroy, NThread %x",
			2 => "EThreadCreate, NThread %x, DProcess %x, name %s",
			3 => "EThreadDestroy, NThread %x, DProcess %x, Id %x",
			4 => "EThreadName, NThread %x, DProcess %x, name %s",
			5 => "EProcessName, NThread %x, DProcess %x, name %s",
			6 => "EThreadId, NThread %x, DProcess %x, Id %x",
			7 => "EProcessCreate, DProcess %x",
			8 => "EProcessDestroy, DProcess %x",
			},
		);

	my @typedefs;
	my @members;
	my %values	= (
#		UTF::KInitialClientFormat		=>	{type=>"TFormatId", size=>2, value=>512}
		KMaxTUint8						=> {type=>"TUint8", size=>1, value=>255},
		KMaxTUint16						=> {type=>"TUint16", size=>2, value=>65535}
	);
	my %macros;
	my @classes;
	my @enums;
	my %formatStrings;		# each enum may have it's own format string
	my %formatCategories;	# each enum may have it's own format category
	
	my %filescope;
	$filescope{file}=1;
	undef $filescope{name};	

	$filescope{typedefs}=\@typedefs;
	$filescope{members}=\@members;
	$filescope{values}=\%values;
	$filescope{macros} = \%macros;
	$filescope{FormatTables} = \%FormatTables;
	
	$filescope{classes} = \@classes;
	$filescope{enums} = \@enums;

	$filescope{formatStrings} =\%formatStrings;
	$filescope{formatCategories} = \%formatCategories;
	
		
		
	if (@ARGV == 0)
  		{
  		print "BTraceVw.pl \n";
  		print "An unsupported utility which extracts UTrace-style format-strings\n";
  		print "from header files & uses these to decode a BTrace output file\n";
  		print "Syntax : BTraceVw.pl [-v] [-r] [-sd] [-i <IncFilePath>] [<BTrace file>]\n";
  		print "where  : -v  = verbose mode\n";
  		print "       : -r  = raw output mode\n";
  		print "       : -sd = produce SAW trace viewer dictionary file\n";
  		print "       :       this file then needs to be merged into the 'com.symbian.analysis.trace.ui.prefs' file\n";
  		print "       :       located under the carbide workspace directory\n";
		print "\n";
  		
		print "e.g. (this decodes a trace file & produces a comma-separated output file) : \n";
		print "btracevw.pl -i /os/kernelhwsrv/userlibandfileserver/fileserver/inc/f32tracedef.h -i /os/kernelhwsrv/userlibandfileserver/fileserver/inc/utraceefsrv.h -i /os/kernelhwsrv/userlibandfileserver/fileserver/inc/utraceefile.h trace.utf >trace.csv\n";
		print "\n";
		print "e.g. (this overwrites the SAW dictioany file) : \n";
		print "btracevw.pl -sd -i /os/kernelhwsrv/userlibandfileserver/fileserver/inc/f32tracedef.h -i /os/kernelhwsrv/userlibandfileserver/fileserver/inc/utraceefsrv.h -i /os/kernelhwsrv/userlibandfileserver/fileserver/inc/utraceefile.h >com.symbian.analysis.trace.ui.prefs\n";
  		
		exit;
		}

	while (@ARGV > 0)
		{
		
		if ($ARGV[0] eq "-i")
	        {
	        shift @ARGV;
		    ($FilePath) = @ARGV;
	        shift @ARGV;

	        undef @incFiles;
		    @incFiles;
		
		    find sub { push @incFiles, $File::Find::name if m/\.h$/i;}, $FilePath ;
		    foreach $incFile (@incFiles)
		        {
				H2Trace($incFile, \%filescope);
		        }
	        }
		elsif ($ARGV[0] eq "-r")
	        {
		    $RawMode = 1;
   	        shift @ARGV;
	        }
		elsif ($ARGV[0] eq "-sd")
	        {
		    $OutputSawDictionaryMode = 1;
   	        shift @ARGV;
	        }
		elsif ($ARGV[0] eq "-v")
	        {
		    $VerboseMode = 1;
   	        shift @ARGV;
	        }
	    else
	    	{
			$TraceFileName = "$ARGV[0]";
	        shift @ARGV;
	    	}
        }
		
	if ($VerboseMode)
		{
		dump_scope(\%filescope);
		PrintFormatTables(\%FormatTables);
		}
	if ($OutputSawDictionaryMode)
		{
		OutputSawDictionary(\%FormatTables);
		}

    if (defined ($TraceFileName))
        {
        ReadTraceFile($RawMode);
        }

        
        
        
sub ReadTraceFile($)
    {
	(my $RawMode) = @_;
#	print "Trace file is $TraceFileName, RawMode $RawMode, VerboseMode $VerboseMode\n\n";

	open (LOGFILE, "<$TraceFileName") or die "Can't open $TraceFileName: $!\n";
	binmode (LOGFILE);

	my $val = 0;


	# enum TFlags from e32btrace.h
	$EHeader2Present	= 1<<0;
	$ETimestampPresent	= 1<<1;
	$ETimestamp2Present	= 1<<2;
	$EContextIdPresent	= 1<<3;
	$EPcPresent			= 1<<4;
	$EExtraPresent		= 1<<5;
	$ERecordTruncated	= 1<<6;
	$EMissingRecord		= 1<<7;
	
	# enum TFlags2 from e32btrace.h
	$EMultipartFlagMask	= 3<<0;
	$ECpuIdMask			= 0xfff<<20;

	# enum TMultiPart from e32btrace.h
	$EMultipartFirst	= 1;
	$EMultipartMiddle	= 2;
	$EMultipartLast		= 3;
	
	$EMaxBTraceDataArray = 80;
	
	# enum TCategory from e32btrace.h
	$EThreadIdentification = 3;
	
	# enum TThreadIdentification from e32btrace.h
	$EThreadCreate = 2;
	$EThreadName = 4;
	$EProcessName = 5;
	$EThreadId = 6;
	
	# Context Id bits from e32btrace.h
	$EContextIdMask = 0x00000003;
	$EContextIdThread = 0;
	$EContextIdFIQ = 0x1;
	$EContextIdIRQ = 0x2;
	$EContextIdIDFC = 0x3;

	# enum TClassificationRange from e32btraceu.h
	$EAllRangeFirst = 192;
	$EAllRangeLast = 222;

	%TCategoryIdToString = 
		(
		0 => "ERDebugPrintf",
		1 => "EKernPrintf",
		2 => "EPlatsecPrintf",
		3 => "EThreadIdentification",
		4 => "ECpuUsage",
        5 => "EKernPerfLog",
        6 => "EClientServer",
        7 => "ERequests",
        8 => "EChunks",
        9 => "ECodeSegs",
		10 => "EPaging",
		11 => "EThreadPriority",
		12 => "EPagingMedia",
		13 => "EKernelMemory",
		14 => "EHeap",
		15 => "EMetaTrace",
		16 => "ERamAllocator",
		17 => "EFastMutex",
		18 => "EProfiling", 
        19 => "EResourceManager",
        20 => "EResourceManagerUs",
		21 => "ERawEvent ",
		128 => "EPlatformSpecificFirst",
		191 => "EPlatformSpecificLast",
		192 => "ESymbianExtentionsFirst",

		# UTrace "ALL" range 
		192 => "EPanic",
		193 => "EError",
		194 => "EWarning", 
		195 => "EBorder", 
		196 => "EState", 
		197 => "EInternals", 
		198 => "EDump", 
		199 => "EFlow", 
		200 => "ESystemCharacteristicMetrics", 
		201 => "EAdhoc",

		253 => "ESymbianExtentionsLast",
		254 => "ETest1",
		255 => "ETest2",
		);


	%ProcessNames;
	%ThreadNames;
	%ThreadIds;
	
	
	# print column titles
	if ($PrintFlagFilePos) {printf "FilePos, ";}	# col #0
	if ($PrintFlagHdrLen) {	printf "Len, ";}		# col #1
	if ($PrintFlagHdrFlags) {printf "Flags, "; }	# col #2
	printf "Category, ";			# col #3
	printf "TimeStamp, ";			# col #4
	printf "Delta, ";				# col #5
	printf "context Id, ";			# col #6
	printf "PC, ";					# col #7
	printf "UID, ";					# col #8
	if ($PrintFlagFormatString){printf "Format string, ";}	# col #9
	printf "Formatted text, ";		# col #10
	print "\n\n";

	
	while (1)
		{
		my $pos = tell (LOGFILE);
		
		# print file pos (col #0)
		if ($PrintFlagFilePos){	printf ("0x%08X, ", $pos);}
		
		my $category;
		my $subCategory;
		my $multipartFlags = 0;
		my $recordData = "";
		my $recordLen;
		my $recordPos = 0;
		
		$recordLen = ReadRecord(LOGFILE, \$pos, \$recordData, \$category, \$subCategory, \$multipartFlags, $RawMode);
		if ($recordLen == -1)
			{last;}

			
		if (!$RawMode && ($multipartFlags == $EMultipartMiddle || $multipartFlags == $EMultipartLast))
			{next;}
					
#		print record contents
#		my $buf;
#					for (my $i=0; $i < $recordLen; $i+=4)
#						{
#		$buf.= sprintf ("%08X ", unpack("V", substr($recordData, $recordPos+$i, 4)));
#						}
#		printf "\n[$buf\n]";				


		# for UTrace "ALL" range, read UID 
		if ($category >= $EAllRangeFirst && $category <= $EAllRangeLast && 
			(!$RawMode) && $multipartFlags != $EMultipartMiddle && $multipartFlags != $EMultipartLast)
			{
			$uid = unpack("V", substr($recordData, $recordPos, 4));
			$recordPos+= 4;	

			# then read formatID			
			$FormatIdIsSubCategory = ($subCategory != 0) ? 1 : 0;
			if ($FormatIdIsSubCategory)
				{
				$formatId = $subCategory
				}
			else				
				{
				$formatId = unpack("V", substr($recordData, $recordPos, 4));
  				$recordPos+= 4;
				}
			}
		
					
		# print UID (col #8)
		printf "0x%08X, ", $uid;

			
		my $formatTable;
		my $formatString;
		if ($category >= $EAllRangeFirst && $category <= $EAllRangeLast)
			{
			$formatString = $FormatTables{$uid}{$formatId};
			}
		else
			{
			$formatString = $FormatTables{$category}{$subCategory};
			}


		# Get thread names
		if ($category == $EThreadIdentification)
			{
			if ($subCategory == $EProcessName)
				{
				my $process = unpack("V", substr($recordData, 4, 4));
				my $processName = substr($recordData, 8, $recordLen - 8);	
#				printf ("\nprocess [%08X] processName [$processName]\n", $process);
				$ProcessNames{$process} = $processName;
				}
			elsif ($subCategory == $EThreadCreate || $subCategory == $EThreadName)
				{
				my $thread = unpack("V", substr($recordData, 0, 4));
				my $process = unpack("V", substr($recordData, 4, 4));
				my $threadName = substr($recordData, 8, $recordLen - 8);	
#				printf ("\nprocess [%08X] thread [%08X] threadName [$threadName]\n", $process, $thread, $threadName);
				$ThreadNames{$thread} = $ProcessNames{$process} . "::" . $threadName;
				}
			elsif ($subCategory == $EThreadId)
				{
				my $thread = unpack("V", substr($recordData, 0, 4));
				my $process = unpack("V", substr($recordData, 4, 4));
				my $threadId = unpack("V", substr($recordData, 8, 4));
#				printf ("\nprocess [%08X] thread [%08X] threadId [%08X]\n", $process, $thread, $threadId);
				$ThreadIds{$thread} = $threadId;
				}
			}
			
			
		# print Format string (col #9)
		if ($PrintFlagFormatString)
			{
			my $formatStringWithoutCommas = $formatString;
			$formatStringWithoutCommas=~ s/,/ /g;
			printf "%s, ", $formatStringWithoutCommas;
			}

		my $formattedText;
		
		my $lenFormatString = length($formatString);
		if ($lenFormatString && !$RawMode && $multipartFlags != $EMultipartMiddle && $multipartFlags != $EMultipartLast)
			{
			for (my $i=0; $i<$lenFormatString; $i++)
				{
				my $c = (substr ($formatString, $i, 1));
#				printf "$c\n";
				if ($c eq "%")
					{
					undef my $fieldLen;
					$i++;
	        		$c = (substr ($formatString, $i, 1));
					if ($c eq "%")
						{
						$formattedText.= substr ($formatString, $i, 1);
						next;
						}
					if ($c eq "*")	## take length from buffer
						{
						$fieldLen = unpack("V", substr($recordData, $recordPos, 4));
						if ($fieldLen > $recordLen-$recordPos)
							{
							$formattedText.= "*** Invalid field length ***";
							last;
							}
						$recordPos+= 4;
						$i++;
		        		$c = (substr ($formatString, $i, 1));
						}
					if (lc $c eq "x" || $c eq "h")
						{
						if (defined $fieldLen)
							{
							if (($fieldLen & 3) == 0)
								{
								for (my $i=0; $i< $fieldLen; $i+= 4)
									{
									$formattedText.= sprintf ("%08X ", unpack("V", substr($recordData, $recordPos, 4)));
									$recordPos+= 4;
									}
								}
							else
								{
								for (my $i=0; $i< $fieldLen; $i++)
									{
									$formattedText.= sprintf ("%02X ", unpack("C", substr($recordData, $recordPos, 1)));
									$recordPos++;
									}
								}
							}
						else
							{
							$formattedText.= sprintf ("0x%08X", unpack("V", substr($recordData, $recordPos, 4)));
							$recordPos+= 4;
							}
						$recordPos = ($recordPos + 3) & ~3;
						next;
						}
					# display "%ld" as hex for now as don't know how to get perl to use or display a 64 decimal value
					elsif (lc $c eq "l" && substr ($formatString, $i+1, 1) eq "d")
						{
						$i++;
						my $loWord = unpack("V", substr($recordData, $recordPos, 4));
						$recordPos+= 4;
						my $hiWord = unpack("V", substr($recordData, $recordPos, 4));
						$recordPos+= 4;
						$formattedText.= sprintf ("0x%X:%08X", $hiWord, $loWord);
						}
					elsif (lc $c eq "l" && substr ($formatString, $i+1, 1) eq "x")
						{
						$i++;
						my $loWord = unpack("V", substr($recordData, $recordPos, 4));
						$recordPos+= 4;
						my $hiWord = unpack("V", substr($recordData, $recordPos, 4));
						$recordPos+= 4;
						$formattedText.= sprintf ("0x%X:%08X", $hiWord, $loWord);
						}
					elsif (lc $c eq "d")
						{
						$formattedText.= sprintf ("%d", unpack("V", substr($recordData, $recordPos, 4)));
						$recordPos+= 4;
						$recordPos = ($recordPos + 3) & ~3;
						next;
						}
					elsif ($c eq "s")
						{
						if (!defined $fieldLen) 
							{$fieldLen = $recordLen - $recordPos;}
						$formattedText.= substr($recordData, $recordPos, $fieldLen);
						$recordPos+= $fieldLen; 
						$recordPos = ($recordPos + 3) & ~3;
						next;
						}
					elsif ($c eq "S")
						{
						if (!defined $fieldLen) 
							{$fieldLen = $recordLen-$recordPos;}
						for (my $j=0; $j < $fieldLen; $j+=2)
							{
					        my $byte = unpack("c", substr ($recordData, $recordPos+$j, 1));
 							$formattedText.= sprintf ("%c", $byte);
							}
						$recordPos+= $fieldLen; 
						$recordPos = ($recordPos + 3) & ~3;
						next;
						}
					elsif ($c eq "c")
						{
				        my $byte = unpack("c", substr ($recordData, $recordPos, 1));
						$formattedText.= sprintf ("%c", $byte);
						}
					}
				else
					{
					$formattedText.= $c;
					}
				}
			}
		else	# no format string : print as hex
			{
			for (my $i=0; $i < $recordLen; $i+=4)
				{
				$formattedText.= sprintf ("%08X ", unpack("V", substr($recordData, $i, 4)));
				}
			$recordPos+= $recordLen; $recordLen = 0;
			
			}
		

		# print Formatted text (col #10)
		$formattedText=~ s/,/;/g;
		$formattedText=~ s/\r//g;
		$formattedText=~ s/\n/,/g;
		printf "%s", $formattedText;

		printf("\n");

		if ($len < 0 || $recordLen < 0)	{die "truncated file";}
  

		$pos+= ($len +3) & ~3;
		seek (LOGFILE, $pos, SEEK_SET) or die "truncated file";
		$i++;
		}

	close (LOGFILE);

	if ($VerboseMode)
		{
		print "*** Processes ***\n";
		for $id ( keys %ProcessNames )
			{
			printf ("process %08X ProcessName %s\n", $id, $ProcessNames{$id});
			}
		print "*** Thread ***\n";
		for $id ( keys %ThreadNames )
			{
			printf ("thread %08X ThreadName %s::%X\n", $id, $ThreadNames{$id}, $ThreadIds{$id});
			}
		}

    }

    
sub ReadSingleRecord
	{
	($fh, $data, $dataLen, $recordLen, $category, $subCategory, $multipartFlags, $extraN, $totalLen, $offset, $RawMode) = @_;	
	
	my $hdr;
	my $flags;
	my $header2;
	my $timestamp;
	my $timestamp2;
	my $contextId;
	my $programConter;	
	
	my $recordOffset = 0;
	
	$timestampLast;	
	my $timestampDelta = 0;	
	
	my $bytesRead = read($fh, $hdr, 4);
	
	
	if ($bytesRead < 4)	
		{return -1;}

	($$recordLen,$flags,$$category,$$subCategory) = unpack("CCCC", $hdr);
	$$dataLen = $$recordLen-4;
	
	if ($flags & $EHeader2Present)
		{$$multipartFlags = (ReadDword($fh) & $EMultipartFlagMask); $$dataLen-= 4}
	else
		{$$multipartFlags = 0;}
	if ($flags & $ETimestampPresent)
		{$timestamp = ReadDword($fh); $$dataLen-= 4;}
	if ($flags & $ETimestamp2Present)
		{$timestamp2 = ReadDword($fh); $$dataLen-= 4;}
	if ($flags & $EContextIdPresent)
		{$contextId = ReadDword($fh); $$dataLen-= 4;}
	if ($flags & $EPcPresent)
		{$programConter = ReadDword($fh); $$dataLen-= 4;}
	if ($flags & $EExtraPresent)
		{$$extraN = ReadDword($fh); $$dataLen-= 4;}
	if ($$multipartFlags != 0)
		{
		$$totalLen = ReadDword($fh);  $$dataLen-= 4;
		if ($$multipartFlags == $EMultipartMiddle || $$multipartFlags == $EMultipartLast)
			{$$offset = ReadDword($fh);  $$totalLen-= 4; $$dataLen-= 4;}
		}				

	$timestampDelta = $timestamp - $timestampLast;
	$timestampLast = $timestamp;

	read($fh, $$data, ($$dataLen + 3) & ~3);


	if ($RawMode || $$multipartFlags == $EMultipartFirst || $$multipartFlags == 0)
		{
		# print header len (col #1)
		if ($PrintFlagHdrLen){printf ("0x%02X, ", $$recordLen);}
	
		# print header flags (col #2)
		if ($PrintFlagHdrFlags)
			{
			printf ("%02X ", $flags);
			if ($flags & $EHeader2Present) {printf "EHeader2Present ";}
			if ($flags & $ETimestampPresent) {printf "ETimestampPresent ";}
			if ($flags & $ETimestamp2Present) {printf "ETimestamp2Present ";}
			if ($flags & $EContextIdPresent) {printf "EContextIdPresent ";}
			if ($flags & $EPcPresent) {printf "EPcPresent ";}
			if ($$multipartFlags != 0)
				{
				printf "EExtraPresent ";
				if ($$multipartFlags == $EMultipartFirst) {print "EMultipartFirst ";}
				elsif ($$multipartFlags == $EMultipartMiddle) {print "EMultipartMiddle ";}
				elsif ($$multipartFlags == $EMultipartLast) {print "EMultipartLast ";}
				printf ("ExtraN(0x%08X) ", $$extraN);
				}
			if ($flags & $ERecordTruncated) {printf "ERecordTruncated ";}
			if ($flags & $EMissingRecord) {printf "EMissingRecord ";}
			print ",";
			}
				
		# print category (col #3)
		printf "(%d;%d) $categoryString  , ", $$category, $$subCategory;
	
		# print timestamp(s) (col #4)
		printf "0x";
		if (defined $timestamp2) {printf "%08X : ", $timestamp2;}
		printf "%08X", $timestamp;
		printf ", ";;
	
		# print timestamp delta (col #5)
		printf "0x%08X, ", $timestampDelta;

		# print context Id (col #6)
		if (!$RawMode && defined $ThreadNames{$contextId})
			{
			printf ("%s::%X, ", $ThreadNames{$contextId}, $ThreadIds{$contextId});
			}
		else			
			{
			if ((($contextId & $EContextIdMask) == $EContextIdThread) || $RawMode)
				{printf "0x%08X, ", $contextId;}
			elsif (($contextId & $EContextIdMask) == $EContextIdFIQ)
				{printf "FIQ, ";}
			elsif (($contextId & $EContextIdMask) == $EContextIdIRQ)
				{printf "IRQ, ";}
			elsif (($contextId & $EContextIdMask) == $EContextIdIDFC)
				{printf "IDFC, ";}
			}
	
		# print Program Counter (col #7)
		printf "0x%08X, ", $programConter;
		}

		
	
	
#########################################################
#	my $hex;
#	for (my $i=0; $i < $$dataLen; $i+=4)
#		{
#		$hex.= sprintf ("%08X ", unpack("V", substr($$data, $i, 4)));
#		}
#	printf "\nadding [$hex]\n";
#########################################################
	return $bytesRead
	}

	      
sub ReadRecord 
	{
	($fh, $recordPos, $recordData, $category, $subCategory, $multipartFlags, $RawMode) = @_;
#	printf "CurrentPos %08X\n", $pos;



	seek ($fh, $$recordPos, SEEK_SET) or die "truncated file";
	my $recordLen;
	my $extraN;
	my $totalLen;
	my $offset;
	my $dataLen;
	my $data;
	my $bytesRead;
	
	
	$bytesRead = ReadSingleRecord($fh,  \$data, \$dataLen, \$recordLen, \$$category, \$$subCategory, \$$multipartFlags, \$extraN, \$totalLen, \$offset, $RawMode);

	if ($bytesRead == -1)	# eof ?
		{return -1; }
	$$recordPos+= ($recordLen +3) & ~3;
	
	$$recordData = $data;
    $offset = $dataLen;

	$offset-= 4;		# subtract 4 bytes for UID ?????????
    
    if ($RawMode || $$multipartFlags != $EMultipartFirst)
    	{return $dataLen;}

    $pos = $$recordPos;

	while (1)
		{
		
		# find next record, i.e. look for a record which matches $extraN 
		
		seek ($fh, $pos, SEEK_SET) or die "truncated file";

		my $recordLen;
		
		my $category;
		my $subCategory;
		my $multipartFlags;
		my $currentExtraN;
		my $currentOffset;
		
		my $totalLen;
		my $currentDataLen;
		my $data;
		$bytesRead = ReadSingleRecord($fh, \$data, \$currentDataLen, \$recordLen, \$category, \$subCategory, \$multipartFlags, \$currentExtraN, \$totalLen, \$currentOffset, $RawMode);
		if ($bytesRead == -1)	# eof ?
			{return -1; }
		$pos+= ($recordLen +3) & ~3;
		
#		printf "\npos %08X, Seaching for (extra %08X, offset %08X), found (extra %08X, offset %08X)\n",
#			$pos, $extraN, $offset, $currentExtraN, $currentOffset;

		if ($currentExtraN == $extraN && $currentOffset == $offset)
			{
			$$recordData.= $data;
			$offset+= $currentDataLen;
			$dataLen+= $currentDataLen;
			}
			
		if ($multipartFlags == $EMultipartLast)
			{last;}
		}
	
	return $dataLen;
	}	

sub ReadDword {
	(my $fh) = @_;
	my $buffer;

	$bytesRead = read($fh, $buffer, 4);
	if ($bytesRead < 4) 	{die "truncated file";}

	my $dword = unpack("V", $buffer);

	return $dword
	};

sub ReadByte {
	(my $fh) = @_;
	my $buffer;

	$bytesRead = read($fh, $buffer, 1);
	if ($bytesRead < 1) 	{die "truncated file";}

	my $byte = unpack("C", $buffer);

	return $byte
	};

    
	
sub PrintFormatTables($)
	{
	my ($formatTables) = @_;
		
	for $tableIndex ( sort keys %$formatTables )
		{
		printf ("SYMTraceFormatCategory %08X:\n", $tableIndex);
		for $formatId (sort keys %{ $$formatTables{$tableIndex} } )
			{
			printf ("%08X => %s\n", $formatId, $$formatTables{$tableIndex}{$formatId});
			}
			print "\n";
		}
	}
        


sub OutputSawDictionary($)
	{
	my ($formatTables) = @_;


	# SAW enums
	$EFieldTypeHexDump = 0;
	$EFieldTypeHex = 1;
	$EFieldTypeDecimal = 2;
	$EFieldTypeStringToEnd = 3;
	$EFieldTypeNullTerminatedString = 4;
	$EFieldTypeHexDumpToEnd = 5;
	$EFieldTypeUnicodeToEnd = 6;
	$EFieldTypeNullTerminatedUnicode = 7;
	$EFieldTypeCountedUnicode = 8;
	$EFieldTypeCountedHexDump = 9;
	$EFieldTypeCountedString = 10;

	my $moduleIds;	# string containg all UIDs separared by semi-colons
		
	for $tableIndex ( sort keys %$formatTables )
		{
		if ($tableIndex < 256)
			{
			next;
			}
		$moduleIds.= sprintf ("%08X;", $tableIndex);
		
		printf ("MODULEID_%08X_DESC=\n", $tableIndex);
		printf ("MODULEID_%08X_NAME=%08X\n", $tableIndex, $tableIndex);
		
		my $formatIds;
		$formatIds = sprintf ("MODULEID_%08X_FORMATIDS=", $tableIndex);
		
		for $formatId  (sort keys %{ $$formatTables{$tableIndex} } )
			{
			$formatIds.= sprintf ("%d;", $formatId);
			}
		printf ("$formatIds\n");
		
		
		for $formatId (sort keys %{ $$formatTables{$tableIndex} } )
			{
			my $fieldCount = 0;
			my $formatString = $$formatTables{$tableIndex}{$formatId};
			
#printf ("formatString = (%s)\n", $formatString);

			# format name is the first format string up until the first space or '%' character or end-of line ...
			$formatString=~ m/^[^%\s]*/;
			my $formatName = $&;
			
			# thow the format name away
			$formatString = $';
			
			# strip the leading space
			$formatString=~ s/\s*//;

			printf ("MODULEID_%08X_FORMATID_%d_NAME=%s\n", $tableIndex, $formatId, $formatName);
#printf ("MODULEID_%08X_FORMATID_%d_DESC=\n", $tableIndex, $formatId);

			my $lenFormatString = length($formatString);
			
			my $formattedText;
			my $fieldType = $EFieldTypeHex;
			my $fieldLen = 0;
			while (length($formatString))
				{
				my $c = (substr ($formatString, 0, 1));
#print ("[$formatString][$c]\n");				
				$formatString=~ s/.//;	# strip the leading space
				if ($c eq "%")
					{
#print "found %\n";							
					my $fieldLenSpecified = 0;
	        		$c = (substr ($formatString, 0, 1));
					$formatString=~ s/.//;	# discard char
#print "c2=$c\n";							
					if ($c eq "%")
						{
						$formattedText.= substr ($formatString, 0, 1);
						next;
						}
					if ($c eq "*")	## take length from buffer
						{
						$fieldLenSpecified = 1;
		        		$c = (substr ($formatString, 0, 1));
						$formatString=~ s/.//;	# discard char
						}
					if (lc $c eq "x" || $c eq "h")
						{
						## deal wilth $fieldLenSpecified
						if ($fieldLenSpecified)
							{
							$fieldType = $EFieldTypeCountedHexDump;
							$fieldLen = 0;
							}
						else
							{
							$fieldType = $EFieldTypeHex;
							$fieldLen = 4;
							}
						}
					elsif (lc $c eq "l" && substr ($formatString, 0, 1) eq "d")
						{
						$formatString=~ s/.//;	# discard char
						$fieldType = $EFieldTypeDecimal;
						$fieldLen = 8;
						}
					elsif (lc $c eq "l" && substr ($formatString, 0, 1) eq "x")
						{
						$formatString=~ s/.//;	# discard char
						$fieldType = $EFieldTypeHex;
						$fieldLen = 8;
						}
					elsif (lc $c eq "d")
						{
						$fieldType = $EFieldTypeDecimal;
						$fieldLen = 4;
						}
					elsif ($c eq "s")
						{
						## deal wilth $fieldLenSpecified
						if ($fieldLenSpecified)
							{
							$fieldType = $EFieldTypeCountedString;
							$fieldLen = 0;
							}
						else
							{
							$fieldType = $EFieldTypeStringToEnd;
							$fieldLen = 0;
							}
						}
					elsif ($c eq "S")
						{
						## deal wilth $fieldLenSpecified
						if ($fieldLenSpecified)
							{
							$fieldType = $EFieldTypeCountedUnicode;
							$fieldLen = 0;
							}
						else
							{
							$fieldType = EFieldTypeUnicodeToEnd;
							$fieldLen = 0;
							}
						}
					elsif ($c eq "c")
						{
						$fieldType = $EFieldTypeHex;
						$fieldLen = 1;
						}
					printf ("MODULEID_%08X_FORMATID_%d_FIELD_%d_NAME=%s\n", $tableIndex, $formatId, $fieldCount, $formattedText);
					printf ("MODULEID_%08X_FORMATID_%d_FIELD_%d_TYPE=%s\n", $tableIndex, $formatId, $fieldCount, $fieldType);
					if ($fieldLen > 0)
						{printf ("MODULEID_%08X_FORMATID_%d_FIELD_%d_LENGTH=%s\n", $tableIndex, $formatId, $fieldCount, $fieldLen);}
					$fieldCount++;
					$formattedText="";
					
					$formatString=~ s/\s//;	# strip the leading space
					}
				else
					{
#					if ($c eq ":") {$formattedText.= '\\'; }
					$formattedText.= $c;
					}
				}
			printf ("MODULEID_%08X_FORMATID_%d_FIELDS=%d\n", $tableIndex, $formatId, $fieldCount);
			
			}
		print "MODULEIDS=$moduleIds\n";
		}
	}
	
	
	
	
	
	
	        
        
sub H2Trace($$)
{
	%basictypes = (
		TInt8		=>	1,
		TUint8		=>	1,
		TInt16		=>	2,
		TUint16		=>	2,
		TInt32		=>	4,
		TUint32		=>	4,
		TInt		=>	4,
		TUint		=>	4,
		TBool		=>	4,
		TInt64		=>	8,
		TUint64		=>	8,
		TLinAddr	=>	4,
		TVersion	=>	4,
		TPde		=>	4,
		TPte		=>	4,
		TProcessPriority => 4,
		TFormatId	=>  2,
	);
	
	if (scalar(@_)!= 2) {
		die "perl h2trace.pl <input.h>\n";
	}
	my ($infile, $filescope) = @_;
	
	if ($VerboseMode)
		{print "\nOpening $infile\n";}
	
	open IN, $infile or die "Can't open $infile for input\n";
	my $in;
	while (<IN>) {
		$in.=$_;
	}
	close IN;
	
	# First remove any backslash-newline combinations
	$in =~ s/\\\n//gms;
	
	# Remove any character constants
	$in =~  s/\'(.?(${0})*?)\'//gms;
	
	# Strip comments beginning with //
	$in =~ s/\/\/(.*?)\n/\n/gms;    #//(.*?)\n
	
	# Strip comments (/* */) but leave doxygen comments (/** */)
	$in =~ s/\/\*[^*](.*?)\*\//\n/gms;  #/*(.*?)*/
	
	
	# Collapse whitespace into a single space or newline
	$in =~ s/\t/\ /gms;
	$in =~ s/\r/\ /gms;
	
	# Tokenize on non-identifier characters
	my @tokens0 = split(/(\W)/,$in);
	my @tokens;
	my $inString = 0;
	my $inComment = 0;
	my $string;
	foreach $t (@tokens0) {
		next if ($t eq "");
		next if (!$inString && ($t eq " " or $t eq ""));
		if ($inComment == 0) 
			{
			if ($t eq "/")
				{$inComment = 1;}
			}
		elsif ($inComment == 1) 
			{
			if ($t eq "*")
				{$inComment = 2;}
			else
				{$inComment = 0;}
			}
		elsif ($inComment == 2) 
			{
			if ($t eq "*")
				{$inComment = 3;}
			}
		elsif ($inComment == 3) 
			{
			if ($t eq "/")
				{
				$inComment = 0;
		        # if we were in a string, need to push previous '*'
		        if ($inString)
		          {
		          push @tokens, "*";
		          }
				$inString = 0;	# end of comment aborts a string
				$string = "";
				}
			else
				{$inComment = 2;}
			}
			
		if ($t eq "\"")
			{
			if (!$inString) 
				{
				$inString=1;
				next;
				}
			else
				{
				$inString=0;
				$t = $string;
				$string = "";
#				if ($VerboseMode) {print "string : [$t]\n";	}
				}
			}
			
		if ($inString)
			{
			$string.= $t;
			next;
			}
		push @tokens, $t;
	}
	
	my $CurrentTraceFormatString;
	my $CurrentTraceFormatCategory;
	# format Key as specified by the @TraceFormatCategory tag is either the current category 
	# or the current UID
	my $CurrentFormatTableKey;	
	
	
	my $line=1;
	parse_scope($filescope, \@tokens, \$line);

	#print $in;
	#print join (" ", @tokens);
}	# end of     H2Trace
	


	sub parse_scope($$$) {
		my ($scope, $tokens, $line) = @_;
		my $state = 1;
		
		my @classes;
		my $curr_offset=0;
		my $overall_align=0;
#		print ">parse_scope $scope->{name}\n";
		
		while (scalar(@$tokens))
			{
			my $t = shift @$tokens;
#			printf "t: [$t] [$$line]\n";
	    	if (!defined ($t)) {
	      		printf "undefined !";
	      		next;
	      	}
			if ($state>=-1 and $t eq "\n") {
				++$$line;
				$state=1;
				next;
			} elsif ($state==-1 and $t ne "\n") {
				next;
			} elsif ($state==-2 and $t ne ';') {
				next;
			}
			
			if ($state>0 and $t eq '#') {
				$t = shift @$tokens;
				if ($t eq 'define') {
					my $ident = shift @$tokens;
					my $defn = shift @$tokens;
					if ($defn ne '(') {	# don't do macros with parameters
#					print "MACRO: $ident :== $defn\n";
					$macros{$ident} = $defn;
					}
				}
				$state=-1;	# skip to next line
				next;
			}
			
			
			if (parse_doxygen($scope,$tokens, $line, $t) == 1)
				{next;}
	
			if ($t eq "namespace" ) {
				$state=0;
				my %cl;
				$cl{specifier}=$t;
				$cl{scope}=$scope;
				$cl{values}=$scope->{values};
				$cl{members}=\$scope->{members};
				$cl{typedefs}=\$scope->{typedefs};
				$cl{FormatTables}=$scope->{FormatTables};
				$cl{formatStrings} =$scope->{formatStrings};
				$cl{formatCategories} =$scope->{formatCategories};
				
				my $new_namespace = \%cl;
				my $n = get_token($scope,$tokens,$line);
				if ($n !~ /\w+/) {
					warn "Unnamed $t not supported at line $$line\n";
					return;
				}
				$new_namespace->{name}=$n;
				my @class_match = grep {$_->{name} eq $n} @classes;
				my $exists = scalar(@class_match);
				my $b = get_token($scope,$tokens,$line);
				if ($b eq ':') {
					die "Inheritance not supported at line $$line\n";
				} elsif ($b eq ';') {
					# forward declaration
					push @classes, $new_namespace unless ($exists);
					next;
				} elsif ($b ne '{') {
					warn "Syntax error#1 at line $$line\n";
					return;
				}
				if ($exists) {
					$new_namespace = $class_match[0];
					if ($new_namespace->{complete}) {
						warn "Duplicate definition of $cl{specifier} $n\n";
					}
				}
				push @classes, $new_namespace unless ($exists);
				parse_scope($new_namespace, $tokens, $line);
				next;
			}
			
			if ($t eq "struct" or $t eq "class" or $t eq "NONSHARABLE_CLASS") {
				next if ($state==0);
				$state=0;
				my %cl;
				$cl{specifier}=$t;
				$cl{scope}=$scope;
				my @members;
				my @typedefs;
				$cl{members}=\@members;
				$cl{typedefs}=\@typedefs;
				$cl{FormatTables}=$scope->{FormatTables};
				my $new_class = \%cl;
				my $n;

				if ($t eq "NONSHARABLE_CLASS")
					{
					my $b = get_token($scope,$tokens,$line);
					if ($b !~ /\(/) {die "Syntax error at line $$line\n";}
					$n = get_token($scope,$tokens,$line);
  				$b = get_token($scope,$tokens,$line);
					if ($b !~ /\)/) {die "Syntax error at line $$line\n";}
					}
				else					
					{
					$n = get_token($scope,$tokens,$line);
					}
								
				
				if ($n !~ /\w+/) {
					warn "Unnamed $t not supported at line $$line\n";
					return;
				}
				$new_class->{name}=$n;
				my @class_match = grep {$_->{name} eq $n} @classes;
				my $exists = scalar(@class_match);
				my $b = get_token($scope,$tokens,$line);
				#skip inheritance etc until we get to a '{' or \ ';'
				while ($b ne '{' && $b ne ';')
					{
			        $b = get_token($scope,$tokens,$line);
			        die "Syntax error#2 at line $$line\n" if  (!defined $b);
					}
				if ($b eq ';') {
					# forward declaration
					push @classes, $new_class unless ($exists);
					next;
				} 
				if ($exists) {
					$new_class = $class_match[0];
					if ($new_class->{complete}) {
						warn "Duplicate definition of $cl{specifier} $n\n";
					}
				}
				push @classes, $new_class unless ($exists);
				parse_scope($new_class, $tokens, $line);
				next;
			} elsif ($t eq "enum") {
				$state=0;
				my $n = get_token($scope,$tokens,$line);
				my $name="";
				if ($n =~ /\w+/) {
					$name = $n;
					$n = get_token($scope,$tokens,$line);
				}
				push @enums, $name;
				if ($n ne '{') {
					die "Syntax error#4 at line $$line\n";
				}
				parse_enum($scope, $tokens, $line, $name);
				next;
			} elsif ($t eq '}') {
				$state=0;
				if ($scope->{scope}) {
			        if ($scope->{specifier} eq "namespace")
			        	{
						$scope->{complete}=1;
#						print "Scope completed\n";
						last;
						}
					$t = get_token($scope,$tokens,$line);
					# skip to next ';'
					while (defined ($t) and $t ne ';')
						{$t = get_token($scope,$tokens,$line);}
					die "Syntax error#5 at line $$line\n" if ($t ne ';');
					$scope->{complete}=1;
#					print "Scope completed\n";
					last;
				}
				warn "Syntax error#5 at line $$line\n";
				return;
			}
			$state=0;
			if ($scope->{scope}) {
				if ($t eq "public" or $t eq "private" or $t eq "protected") {
					if (shift (@$tokens) eq ':') {
						next;	# ignore access specifiers
					}
				die "Syntax error#6 at line $$line\n";
				}
			}
			unshift @$tokens, $t;
			
			my @currdecl = parse_decl_def($scope, $tokens, $line);
#			print scalar (@currdecl), "\n";
			if ($t eq 'static') {
				next;	# skip static members
			}
			my $typedef;
			if ($t eq 'typedef') {
#			print "TYPEDEF\n";
				$typedef = 1;
				$t = shift @currdecl;
				$t = $currdecl[0];
			} else {
#			print "NOT TYPEDEF\n";
				$typedef = 0;
			}
#			print "$currdecl[0]\n";
			next if (scalar(@currdecl)==0);
			
			if ($t eq "const") {
				# check for constant declaration
#				print "CONST $currdecl[1] $currdecl[2] $currdecl[3]\n";
				my $ctype = lookup_type($scope, $currdecl[1]);
#				print "$ctype->{basic}    $ctype->{size}\n";
				if ($ctype->{basic} and $currdecl[2]=~/^\w+$/ and $currdecl[3] eq '=') {
					if ($typedef!=0) {
						die "Syntax error#7 at line $$line\n";
					}
					shift @currdecl;
					shift @currdecl;
					my $type = $ctype->{name};
					my $name;		#### = shift @currdecl;

					if ($scope->{name})
						{	
						$name = $scope->{name} . "::" . shift @currdecl;
						}
					else
						{
						$name = shift @currdecl;
						}
#					printf "[$name,$scope->{name}]";
					my $size = $ctype->{size};
					shift @currdecl;
					my $value = get_constant_expr($scope,\@currdecl,$line);
					$values{$name} = {type=>$type, size=>$size, value=>$value};
					next;
				}
			}
			
			
			
		}
	}
	
	sub get_token($$$) {
		my ($scope,$tokenlist,$line) = @_;
		while (scalar(@$tokenlist)) {
			my $t = shift @$tokenlist;
			return $t if (!defined($t));
			if (parse_doxygen($scope,$tokenlist, $line, $t) == 1)
				{next;}
			if ($t !~ /^[\s]*$/)
				{
				if ($$tokenlist[0] eq ":" and $$tokenlist[1] eq ":")
					{
					$t.= shift @$tokenlist;
					$t.= shift @$tokenlist;
					$t.= shift @$tokenlist;
#					print "Colon-separated token";
					}
				return $t
				}
			++$$line;
		}
  		return undef;
	}
	
	sub skip_qualifiers($) {
		my ($tokens) = @_;
		my $f=0;
		my %quals = (
			EXPORT_C => 1,
			IMPORT_C => 1,
			inline => 1,
			virtual => 0,
			const => 0,
			volatile => 0,
			static => 0,
			extern => 0,
			LOCAL_C => 0,
			LOCAL_D => 0,
			GLDEF_C => 0,
			GLREF_C => 0,
			GLDEF_D => 0,
			GLREF_D => 0
			);
		for (;;) {
			my $t = $$tokens[0];
			my $q = $quals{$t};
			last unless (defined ($q));
			$f |= $q;
			shift @$tokens;
		}
		return $f;
	}
	
	sub parse_indirection($) {
		my ($tokens) = @_;
		my $level = 0;
		for (;;) {
			my $t = $$tokens[0];
			if ($t eq '*') {
				++$level;
				shift @$tokens;
				next;
			}
			last if ($t ne "const" and $t ne "volatile");
			shift @$tokens;
		}
		return $level;
	}
	
	sub get_operand($$$) {
		my ($scope,$tokens,$line) = @_;
		my $t = get_token($scope,$tokens,$line);
		if ($t eq '-') {
			my $x = get_operand($scope,$tokens,$line);
			return -$x;
		} elsif ($t eq '+') {
			my $x = get_operand($scope,$tokens,$line);
			return $x;
		} elsif ($t eq '~') {
			my $x = get_operand($scope,$tokens,$line);
			return ~$x;
		} elsif ($t eq '!') {
			my $x = get_operand($scope,$tokens,$line);
			return $x ? 0 : 1;
		} elsif ($t eq '(') {
			my $x = get_constant_expr($scope,$tokens,$line);
			my $t = get_token($scope,$tokens,$line);
			if ($t ne ')') {
				warn "Missing ) at line $$line\n";
				return undefined;
			}
			return $x;
		} elsif ($t eq "sizeof") {
			my $ident = get_token($scope,$tokens,$line);
			if ($ident eq '(') {
				$ident = get_token($scope,$tokens,$line);
				my $cb = get_token($scope,$tokens,$line);
				if ($cb ne ')') {
					warn "Bad sizeof() syntax at line $$line\n";
					return undefined;
				}
			}
			$ident = look_through_macros($ident);
			if ($ident !~ /^\w+$/) {
				warn "Bad sizeof() syntax at line $$line\n";
				return undefined;
			}
			my $type = lookup_type($scope, $ident);
			if (!defined $type) {
				warn "Unrecognised type $ident at line $$line\n";
				return undefined;
			}
			if ($type->{basic}) {
				return $type->{size};
			} elsif ($type->{enum}) {
				return 4;
			} elsif ($type->{ptr}) {
				return 4;
			} elsif ($type->{fptr}) {
				return 4;
			}
			my $al = $type->{class}->{align};
			my $sz = $type->{class}->{size};
			return ($sz+$al-1)&~($al-1);
		}
		$t = look_through_macros($t);
		if ($t =~ /^0x/i) {
			return oct($t);
		} elsif ($t =~ /^\d/) {
			return $t;
		} elsif ($t =~ /^\w+$/) {
			my $x = lookup_value($scope,$t);
#			die "Unrecognised identifier '$t' at line $$line\n" unless defined($x);
			if (!defined($x)) {
				print "Unrecognised identifier '$t' at line $$line\n" ;
			}
			return $x;
		} elsif ($t =~ /^\w+::\w+$/) {
			my $x = lookup_value($scope,$t);
#			die "Unrecognised identifier '$t' at line $$line\n" unless defined($x);
			if (!defined($x)) {
				print "Unrecognised identifier '$t' at line $$line\n" ;
			}
			return $x;
		} else {
			warn "Syntax error#10 at line $$line\n";
			return undefined;
		}
	}
	
	sub look_through_macros($) {
		my ($ident) = @_;
		while ($ident and $macros{$ident}) {
			$ident = $macros{$ident};
		}
		return $ident;
	}
	
	sub lookup_value($$) {
		my ($scope,$ident) = @_;
		while ($scope) {
			my $vl = $scope->{values};
			if (defined($vl->{$ident})) {
				return $vl->{$ident}->{value};
			}
			$scope = $scope->{scope};
		}
		return undef();
	}
	
	sub lookup_type($$) {
		my ($scope,$ident) = @_;
		if ($basictypes{$ident}) {
			return {scope=>$scope, basic=>1, name=>$ident, size=>$basictypes{$ident} };
		}
		while ($scope) {
			if ($basictypes{$ident}) {
				return {scope=>$scope, basic=>1, name=>$ident, size=>$basictypes{$ident} };
			}
			my $el = $scope->{enums};
			my $cl = $scope->{classes};
			my $td = $scope->{typedefs};
			if (grep {$_ eq $ident} @$el) {
				return {scope=>$scope, enum=>1, name=>$ident, size=>4 };
			}
			my @match_class = (grep {$_->{name} eq $ident} @$cl);
			if (scalar(@match_class)) {
				return {scope=>$scope, class=>$match_class[0]};
			}
			my @match_td = (grep {$_->{name} eq $ident} @$td);
			if (scalar(@match_td)) {
				my $tdr = $match_td[0];
				my $cat = $tdr->{category};
				if ($cat eq 'basic' or $cat eq 'enum' or $cat eq 'class') {
					$ident = $tdr->{alias};
					next;
				} else {
					return { scope=>$scope, $cat=>1, $size=>$tdr->{size} };
				}
			}
			$scope = $scope->{scope};
		}
		return undef();
	}
	
	sub get_mult_expr($$$) {
		my ($scope,$tokens,$line) = @_;
		my $x = get_operand($scope,$tokens,$line);
		my $t;
		for (;;) {
			$t = get_token($scope,$tokens,$line);
			if ($t eq '*') {
				my $y = get_operand($scope,$tokens,$line);
				$x = $x * $y;
			} elsif ($t eq '/') {
				my $y = get_operand($scope,$tokens,$line);
				if ($y != 0)
					{$x = int($x / $y);}
			} elsif ($t eq '%') {
				my $y = get_operand($scope,$tokens,$line);
				if ($y != 0)
					{$x = int($x % $y);}
			} else {
				last;
			}
		}
		unshift @$tokens, $t;
		return $x;
	}
	
	sub get_add_expr($$$) {
		my ($scope,$tokens,$line) = @_;
		my $x = get_mult_expr($scope,$tokens,$line);
		my $t;
		for (;;) {
			$t = get_token($scope,$tokens,$line);
			if ($t eq '+') {
				my $y = get_mult_expr($scope,$tokens,$line);
				$x = $x + $y;
			} elsif ($t eq '-') {
				my $y = get_mult_expr($scope,$tokens,$line);
				$x = $x - $y;
			} else {
				last;
			}
		}
		unshift @$tokens, $t;
		return $x;
	}
	
	sub get_shift_expr($$$) {
		my ($scope,$tokens,$line) = @_;
		my $x = get_add_expr($scope,$tokens,$line);
		my $t, $t2;
		for (;;) {
			$t = get_token($scope,$tokens,$line);
			if ($t eq '<' or $t eq '>') {
				$t2 = get_token($scope,$tokens,$line);
				if ($t2 ne $t) {
					unshift @$tokens, $t2;
					last;
				}
			}
			if ($t eq '<') {
				my $y = get_add_expr($scope,$tokens,$line);
				$x = $x << $y;
			} elsif ($t eq '>') {
				my $y = get_add_expr($scope,$tokens,$line);
				$x = $x >> $y;
			} else {
				last;
			}
		}
		unshift @$tokens, $t;
		return $x;
	}
	
	sub get_and_expr($$$) {
		my ($scope,$tokens,$line) = @_;
		my $x = get_shift_expr($scope,$tokens,$line);
		my $t;
		for (;;) {
			$t = get_token($scope,$tokens,$line);
			if ($t eq '&') {
				my $y = get_shift_expr($scope,$tokens,$line);
				$x = $x & $y;
			} else {
				last;
			}
		}
		unshift @$tokens, $t;
		return $x;
	}
	
	sub get_xor_expr($$$) {
		my ($scope,$tokens,$line) = @_;
		my $x = get_and_expr($scope,$tokens,$line);
		my $t;
		for (;;) {
			$t = get_token($scope,$tokens,$line);
			if ($t eq '^') {
				my $y = get_and_expr($scope,$tokens,$line);
				$x = $x ^ $y;
			} else {
				last;
			}
		}
		unshift @$tokens, $t;
		return $x;
	}
	
	sub get_ior_expr($$$) {
		my ($scope,$tokens,$line) = @_;
		my $x = get_xor_expr($scope,$tokens,$line);
		my $t;
		for (;;) {
			$t = get_token($scope,$tokens,$line);
			if ($t eq '|') {
				my $y = get_xor_expr($scope,$tokens,$line);
				$x = $x | $y;
			} else {
				last;
			}
		}
		unshift @$tokens, $t;
		return $x;
	}
	
	sub get_constant_expr($$$) {
		my ($scope,$tokens,$line) = @_;
		my $x = get_ior_expr($scope,$tokens,$line);
		return $x;
	}
	
	sub parse_enum($$$$) {
		my ($scope,$tokens,$line,$enum_name) = @_;
		my $vl = $scope->{values};
		my $fstr = $scope->{formatStrings};
		my $fcat = $scope->{formatCategories};
		my $fmtTable = $scope->{FormatTables};
		
		my $x = 0;
		for (;;) {
			my $t = get_token($scope,$tokens,$line);
			last if ($t eq '}');
			if (!defined($t)) {
				die "Unexpected end of file #2 at line $$line\n";
			}
			
			if ($t eq '#') {
				next;
				}
			
			if ($t !~ /^\w+$/) {
				warn "Syntax error#11 at line $$line\n";
				next;
			}

			if ($scope->{name})
				{	
				$t = $scope->{name} . "::" . $t;
				}

			if (defined($vl->{$t})) {
				warn "Duplicate identifier [$t] at line $$line\n";
			}
			my $t2 = get_token($scope,$tokens,$line);
			if ($t2 eq ',') {
				$vl->{$t} = {type=>$enum_name, size=>4, value=>$x, enum=>1};
				$fstr->{$t} = $CurrentTraceFormatString; 
				$fcat->{$t} = $CurrentTraceFormatCategory; 
				if (defined $CurrentTraceFormatCategory && defined $CurrentTraceFormatString)
					{ $fmtTable->{$CurrentTraceFormatCategory}{$x} = $CurrentTraceFormatString; }
				undef $CurrentTraceFormatString;
				++$x;
			} elsif ($t2 eq '}') {
				$vl->{$t} = {type=>$enum_name, size=>4, value=>$x, enum=>1};
				$fstr->{$t} = $CurrentTraceFormatString; 
				$fcat->{$t} = $CurrentTraceFormatCategory; 
				if (defined $CurrentTraceFormatCategory && defined $CurrentTraceFormatString)
					{ $fmtTable->{$CurrentTraceFormatCategory}{$x} = $CurrentTraceFormatString; }
				undef $CurrentTraceFormatString;
				++$x;
				last;
			} elsif ($t2 eq '=') {
				$x = get_constant_expr($scope, $tokens, $line);
				$vl->{$t} = {type=>$enum_name, size=>4, value=>$x, enum=>1};
				$fstr->{$t} = $CurrentTraceFormatString; 
				$fcat->{$t} = $CurrentTraceFormatCategory;
				if (defined $CurrentTraceFormatCategory && defined $CurrentTraceFormatString)
					{ $fmtTable->{$CurrentTraceFormatCategory}{$x} = $CurrentTraceFormatString; }
				undef $CurrentTraceFormatString; 
				++$x;
				$t2 = get_token($scope,$tokens,$line);
				last if ($t2 eq '}');
				next if ($t2 eq ',');
				warn "Syntax error#12 at line $$line\n";
			} else {
				unshift @$tokens, $t2;
			}
		}
		my $t = get_token($scope,$tokens,$line);
		if ($t ne ';') {
			warn "Missing ; at line $$line\n";
		}
	}
	
	
	sub  parse_decl_def($$$) {
		my ($scope,$tokens,$line) = @_;
		my $level=0;
		my @decl;
		while ( scalar(@$tokens) ) {
			my $t = get_token($scope,$tokens, $line);
			if ( (!defined ($t) || $t eq ';') and ($level==0)) {
				return @decl;
			}
	
			if ($t eq "static")
				{
				next;
				}
	
			push @decl, $t;
			if ($t eq '{') {
				++$level;
			}
			if ($t eq '}') {
				if ($level==0) {
					warn "Syntax error#13 at line $$line\n";
					unshift @$tokens, $t;
					return @decl;
					
				}
				if (--$level==0) {
					return ();	# end of function definition reached
				}
			}
		}
		die "Unexpected end of file #3 at line $$line\n";
	}
	
	sub dump_scope($) {
		my ($scope) = @_;
		my $el = $scope->{enums};
		my $cl = $scope->{classes};
		my $vl = $scope->{values};
		my $fstr = $scope->{formatStrings};
		my $fcat = $scope->{formatCategories};
		print "SCOPE: $scope->{name}\n";
		if (scalar(@$el)) {
			print "\tenums:\n";
			foreach (@$el) {
				print "\t\t$_\n";
			}
		}
		if (scalar(keys(%$vl))) {
			print "\tvalues:\n";
			foreach $vname (keys(%$vl)) {
				my $v = $vl->{$vname};
				my $x = $v->{value};
				my $t = $v->{type};
				my $sz = $v->{size};
				my $fstring = $fstr->{$vname};
				my $fcategory = $fcat->{$vname};
				if ($v->{enum}) {
					printf ("\t\t$vname\=$x (enum $t) size=$sz fcat=[0x%x] fstr=[%s]\n", $fcategory,$fstring);
				} else {
					printf ("\t\t$vname\=$x (type $t) size=$sz fcat=[0x%x] fstr=[%s]\n", $fcategory, $fstring);
				}
			}
		}
		if ($scope->{scope}) {
			my $members = $scope->{members};
			foreach (@$members) {
				my $n = $_->{name};
				my $sz = $_->{size};
				my $off = $_->{offset};
				my $spc = $_->{spacing};
				if (defined $spc) {
					print "\t$n\[\]\: spacing $spc size $sz offset $off\n";
				} else {
					print "\t$n\: size $sz offset $off\n";
				}
			}
			print "\tOverall size : $scope->{size}\n";
			print "\tOverall align: $scope->{align}\n";
		}
		foreach $s (@$cl) {
			dump_scope($s);
		}
	}
	
	
	
		
	sub parse_doxygen($$$$) {
		my ($scope,$tokens,$line,$t) = @_;
	
		if ($t ne "/")
			{
			return 0;	# not a doxygen comment
			}
		if ($t eq "/") {
			$state=0;
			my $t2 = shift @$tokens;
			my $t3 = shift @$tokens;
	
			if ($t2 ne "*" || $t3 ne "*")
				{
				unshift @$tokens, $t3;
				unshift @$tokens, $t2;
				return 0;	# not a doxygen comment
				}
		}
#		printf "doxygen start on line %d\n", $$line;
		for (;;) {
			my $t = shift @$tokens;
			if (!defined($t)) 
					{
					warn "Unexpected end of file #4 at line $$line\n";	
					return
					}
			
			if ($t eq "\n"){++$$line };
			
			if ($t eq '*')
				{
				my $t2 = shift @$tokens;
				last if ($t2 eq '/');
				unshift @$tokens, $t2;
				}
			
			if ($t eq '@')
				{
				my $t2 = shift @$tokens;
				if ($t2 eq 'SYMTraceFormatString')
					{
					my $t3 = shift @$tokens;
#					if ($VerboseMode){print "SYMTraceFormatString = [$t3]\n";}
					$CurrentTraceFormatString = $t3;
					}
				if ($t2 eq 'SYMTraceFormatCategory')
					{
					$CurrentTraceFormatCategory = get_operand($scope,$tokens,$line);
#					if ($VerboseMode){printf ("SYMTraceFormatCategory = 0x%x\n", $CurrentTraceFormatCategory);}
					}
				else
					{
					unshift @$tokens, $t2;
					}
				}
	
		}
#		printf ("doxygen end  on line %d\n", $$line);
		return 1;	# is a doxygen comment
	}