kerneltest/e32utils/trace/btracevw.pl
branchRCL_3
changeset 43 c1f20ce4abcf
parent 42 a179b74831c9
child 44 3e88ff8f41d5
equal deleted inserted replaced
42:a179b74831c9 43:c1f20ce4abcf
     1 #
       
     2 # Copyright (c) 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 #!/usr/bin/perl
       
    18 
       
    19 use File::Find;
       
    20 use File::Spec::Functions;
       
    21 
       
    22 
       
    23 	my $TraceFileName;
       
    24 	
       
    25 	my $PrintFlagFilePos = 0;
       
    26 	my $PrintFlagHdrLen = 0;
       
    27 	my $PrintFlagHdrFlags = 0;
       
    28 	my $PrintFlagFormatString = 0;
       
    29 	my $VerboseMode = 0;
       
    30 	my $RawMode = 0;
       
    31 	my $FormatIdIsSubCategory = 0;
       
    32 	my $OutputSawDictionaryMode = 0;
       
    33 	
       
    34 	# for the category range 0-191, the format string is indexed by the category & subcategory
       
    35 	%FormatTables = 
       
    36 		(
       
    37 		0 => 			# ERDebugPrintf
       
    38 			{
       
    39 			0 => "ThreadId %h, %s",
       
    40 			},
       
    41 	
       
    42 		1 => 			# ERKernPrintf
       
    43 			{
       
    44 			0 => "ThreadId %h, %s",
       
    45 			},
       
    46 
       
    47 		3 =>			# EThreadIdentification
       
    48 			{	
       
    49 			0 => "ENanoThreadCreate, NThread %x",
       
    50 			1 => "ENanoThreadDestroy, NThread %x",
       
    51 			2 => "EThreadCreate, NThread %x, DProcess %x, name %s",
       
    52 			3 => "EThreadDestroy, NThread %x, DProcess %x, Id %x",
       
    53 			4 => "EThreadName, NThread %x, DProcess %x, name %s",
       
    54 			5 => "EProcessName, NThread %x, DProcess %x, name %s",
       
    55 			6 => "EThreadId, NThread %x, DProcess %x, Id %x",
       
    56 			7 => "EProcessCreate, DProcess %x",
       
    57 			8 => "EProcessDestroy, DProcess %x",
       
    58 			},
       
    59 		);
       
    60 
       
    61 	my @typedefs;
       
    62 	my @members;
       
    63 	my %values	= (
       
    64 #		UTF::KInitialClientFormat		=>	{type=>"TFormatId", size=>2, value=>512}
       
    65 		KMaxTUint8						=> {type=>"TUint8", size=>1, value=>255},
       
    66 		KMaxTUint16						=> {type=>"TUint16", size=>2, value=>65535}
       
    67 	);
       
    68 	my %macros;
       
    69 	my @classes;
       
    70 	my @enums;
       
    71 	my %formatStrings;		# each enum may have it's own format string
       
    72 	my %formatCategories;	# each enum may have it's own format category
       
    73 	
       
    74 	my %filescope;
       
    75 	$filescope{file}=1;
       
    76 	undef $filescope{name};	
       
    77 
       
    78 	$filescope{typedefs}=\@typedefs;
       
    79 	$filescope{members}=\@members;
       
    80 	$filescope{values}=\%values;
       
    81 	$filescope{macros} = \%macros;
       
    82 	$filescope{FormatTables} = \%FormatTables;
       
    83 	
       
    84 	$filescope{classes} = \@classes;
       
    85 	$filescope{enums} = \@enums;
       
    86 
       
    87 	$filescope{formatStrings} =\%formatStrings;
       
    88 	$filescope{formatCategories} = \%formatCategories;
       
    89 	
       
    90 		
       
    91 		
       
    92 	if (@ARGV == 0)
       
    93   		{
       
    94   		print "BTraceVw.pl \n";
       
    95   		print "An unsupported utility which extracts UTrace-style format-strings\n";
       
    96   		print "from header files & uses these to decode a BTrace output file\n";
       
    97   		print "Syntax : BTraceVw.pl [-v] [-r] [-sd] [-i <IncFilePath>] [<BTrace file>]\n";
       
    98   		print "where  : -v  = verbose mode\n";
       
    99   		print "       : -r  = raw output mode\n";
       
   100   		print "       : -sd = produce SAW trace viewer dictionary file\n";
       
   101   		print "       :       this file then needs to be merged into the 'com.symbian.analysis.trace.ui.prefs' file\n";
       
   102   		print "       :       located under the carbide workspace directory\n";
       
   103 		print "\n";
       
   104   		
       
   105 		print "e.g. (this decodes a trace file & produces a comma-separated output file) : \n";
       
   106 		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";
       
   107 		print "\n";
       
   108 		print "e.g. (this overwrites the SAW dictioany file) : \n";
       
   109 		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";
       
   110   		
       
   111 		exit;
       
   112 		}
       
   113 
       
   114 	while (@ARGV > 0)
       
   115 		{
       
   116 		
       
   117 		if ($ARGV[0] eq "-i")
       
   118 	        {
       
   119 	        shift @ARGV;
       
   120 		    ($FilePath) = @ARGV;
       
   121 	        shift @ARGV;
       
   122 
       
   123 	        undef @incFiles;
       
   124 		    @incFiles;
       
   125 		
       
   126 		    find sub { push @incFiles, $File::Find::name if m/\.h$/i;}, $FilePath ;
       
   127 		    foreach $incFile (@incFiles)
       
   128 		        {
       
   129 				H2Trace($incFile, \%filescope);
       
   130 		        }
       
   131 	        }
       
   132 		elsif ($ARGV[0] eq "-r")
       
   133 	        {
       
   134 		    $RawMode = 1;
       
   135    	        shift @ARGV;
       
   136 	        }
       
   137 		elsif ($ARGV[0] eq "-sd")
       
   138 	        {
       
   139 		    $OutputSawDictionaryMode = 1;
       
   140    	        shift @ARGV;
       
   141 	        }
       
   142 		elsif ($ARGV[0] eq "-v")
       
   143 	        {
       
   144 		    $VerboseMode = 1;
       
   145    	        shift @ARGV;
       
   146 	        }
       
   147 	    else
       
   148 	    	{
       
   149 			$TraceFileName = "$ARGV[0]";
       
   150 	        shift @ARGV;
       
   151 	    	}
       
   152         }
       
   153 		
       
   154 	if ($VerboseMode)
       
   155 		{
       
   156 		dump_scope(\%filescope);
       
   157 		PrintFormatTables(\%FormatTables);
       
   158 		}
       
   159 	if ($OutputSawDictionaryMode)
       
   160 		{
       
   161 		OutputSawDictionary(\%FormatTables);
       
   162 		}
       
   163 
       
   164     if (defined ($TraceFileName))
       
   165         {
       
   166         ReadTraceFile($RawMode);
       
   167         }
       
   168 
       
   169         
       
   170         
       
   171         
       
   172 sub ReadTraceFile($)
       
   173     {
       
   174 	(my $RawMode) = @_;
       
   175 #	print "Trace file is $TraceFileName, RawMode $RawMode, VerboseMode $VerboseMode\n\n";
       
   176 
       
   177 	open (LOGFILE, "<$TraceFileName") or die "Can't open $TraceFileName: $!\n";
       
   178 	binmode (LOGFILE);
       
   179 
       
   180 	my $val = 0;
       
   181 
       
   182 
       
   183 	# enum TFlags from e32btrace.h
       
   184 	$EHeader2Present	= 1<<0;
       
   185 	$ETimestampPresent	= 1<<1;
       
   186 	$ETimestamp2Present	= 1<<2;
       
   187 	$EContextIdPresent	= 1<<3;
       
   188 	$EPcPresent			= 1<<4;
       
   189 	$EExtraPresent		= 1<<5;
       
   190 	$ERecordTruncated	= 1<<6;
       
   191 	$EMissingRecord		= 1<<7;
       
   192 	
       
   193 	# enum TFlags2 from e32btrace.h
       
   194 	$EMultipartFlagMask	= 3<<0;
       
   195 	$ECpuIdMask			= 0xfff<<20;
       
   196 
       
   197 	# enum TMultiPart from e32btrace.h
       
   198 	$EMultipartFirst	= 1;
       
   199 	$EMultipartMiddle	= 2;
       
   200 	$EMultipartLast		= 3;
       
   201 	
       
   202 	$EMaxBTraceDataArray = 80;
       
   203 	
       
   204 	# enum TCategory from e32btrace.h
       
   205 	$EThreadIdentification = 3;
       
   206 	
       
   207 	# enum TThreadIdentification from e32btrace.h
       
   208 	$EThreadCreate = 2;
       
   209 	$EThreadName = 4;
       
   210 	$EProcessName = 5;
       
   211 	$EThreadId = 6;
       
   212 	
       
   213 	# Context Id bits from e32btrace.h
       
   214 	$EContextIdMask = 0x00000003;
       
   215 	$EContextIdThread = 0;
       
   216 	$EContextIdFIQ = 0x1;
       
   217 	$EContextIdIRQ = 0x2;
       
   218 	$EContextIdIDFC = 0x3;
       
   219 
       
   220 	# enum TClassificationRange from e32btraceu.h
       
   221 	$EAllRangeFirst = 192;
       
   222 	$EAllRangeLast = 222;
       
   223 
       
   224 	%TCategoryIdToString = 
       
   225 		(
       
   226 		0 => "ERDebugPrintf",
       
   227 		1 => "EKernPrintf",
       
   228 		2 => "EPlatsecPrintf",
       
   229 		3 => "EThreadIdentification",
       
   230 		4 => "ECpuUsage",
       
   231         5 => "EKernPerfLog",
       
   232         6 => "EClientServer",
       
   233         7 => "ERequests",
       
   234         8 => "EChunks",
       
   235         9 => "ECodeSegs",
       
   236 		10 => "EPaging",
       
   237 		11 => "EThreadPriority",
       
   238 		12 => "EPagingMedia",
       
   239 		13 => "EKernelMemory",
       
   240 		14 => "EHeap",
       
   241 		15 => "EMetaTrace",
       
   242 		16 => "ERamAllocator",
       
   243 		17 => "EFastMutex",
       
   244 		18 => "EProfiling", 
       
   245         19 => "EResourceManager",
       
   246         20 => "EResourceManagerUs",
       
   247 		21 => "ERawEvent ",
       
   248 		128 => "EPlatformSpecificFirst",
       
   249 		191 => "EPlatformSpecificLast",
       
   250 		192 => "ESymbianExtentionsFirst",
       
   251 
       
   252 		# UTrace "ALL" range 
       
   253 		192 => "EPanic",
       
   254 		193 => "EError",
       
   255 		194 => "EWarning", 
       
   256 		195 => "EBorder", 
       
   257 		196 => "EState", 
       
   258 		197 => "EInternals", 
       
   259 		198 => "EDump", 
       
   260 		199 => "EFlow", 
       
   261 		200 => "ESystemCharacteristicMetrics", 
       
   262 		201 => "EAdhoc",
       
   263 
       
   264 		253 => "ESymbianExtentionsLast",
       
   265 		254 => "ETest1",
       
   266 		255 => "ETest2",
       
   267 		);
       
   268 
       
   269 
       
   270 	%ProcessNames;
       
   271 	%ThreadNames;
       
   272 	%ThreadIds;
       
   273 	
       
   274 	
       
   275 	# print column titles
       
   276 	if ($PrintFlagFilePos) {printf "FilePos, ";}	# col #0
       
   277 	if ($PrintFlagHdrLen) {	printf "Len, ";}		# col #1
       
   278 	if ($PrintFlagHdrFlags) {printf "Flags, "; }	# col #2
       
   279 	printf "Category, ";			# col #3
       
   280 	printf "TimeStamp, ";			# col #4
       
   281 	printf "Delta, ";				# col #5
       
   282 	printf "context Id, ";			# col #6
       
   283 	printf "PC, ";					# col #7
       
   284 	printf "UID, ";					# col #8
       
   285 	if ($PrintFlagFormatString){printf "Format string, ";}	# col #9
       
   286 	printf "Formatted text, ";		# col #10
       
   287 	print "\n\n";
       
   288 
       
   289 	
       
   290 	while (1)
       
   291 		{
       
   292 		my $pos = tell (LOGFILE);
       
   293 		
       
   294 		# print file pos (col #0)
       
   295 		if ($PrintFlagFilePos){	printf ("0x%08X, ", $pos);}
       
   296 		
       
   297 		my $category;
       
   298 		my $subCategory;
       
   299 		my $multipartFlags = 0;
       
   300 		my $recordData = "";
       
   301 		my $recordLen;
       
   302 		my $recordPos = 0;
       
   303 		
       
   304 		$recordLen = ReadRecord(LOGFILE, \$pos, \$recordData, \$category, \$subCategory, \$multipartFlags, $RawMode);
       
   305 		if ($recordLen == -1)
       
   306 			{last;}
       
   307 
       
   308 			
       
   309 		if (!$RawMode && ($multipartFlags == $EMultipartMiddle || $multipartFlags == $EMultipartLast))
       
   310 			{next;}
       
   311 					
       
   312 #		print record contents
       
   313 #		my $buf;
       
   314 #					for (my $i=0; $i < $recordLen; $i+=4)
       
   315 #						{
       
   316 #		$buf.= sprintf ("%08X ", unpack("V", substr($recordData, $recordPos+$i, 4)));
       
   317 #						}
       
   318 #		printf "\n[$buf\n]";				
       
   319 
       
   320 
       
   321 		# for UTrace "ALL" range, read UID 
       
   322 		if ($category >= $EAllRangeFirst && $category <= $EAllRangeLast && 
       
   323 			(!$RawMode) && $multipartFlags != $EMultipartMiddle && $multipartFlags != $EMultipartLast)
       
   324 			{
       
   325 			$uid = unpack("V", substr($recordData, $recordPos, 4));
       
   326 			$recordPos+= 4;	
       
   327 
       
   328 			# then read formatID			
       
   329 			$FormatIdIsSubCategory = ($subCategory != 0) ? 1 : 0;
       
   330 			if ($FormatIdIsSubCategory)
       
   331 				{
       
   332 				$formatId = $subCategory
       
   333 				}
       
   334 			else				
       
   335 				{
       
   336 				$formatId = unpack("V", substr($recordData, $recordPos, 4));
       
   337   				$recordPos+= 4;
       
   338 				}
       
   339 			}
       
   340 		
       
   341 					
       
   342 		# print UID (col #8)
       
   343 		printf "0x%08X, ", $uid;
       
   344 
       
   345 			
       
   346 		my $formatTable;
       
   347 		my $formatString;
       
   348 		if ($category >= $EAllRangeFirst && $category <= $EAllRangeLast)
       
   349 			{
       
   350 			$formatString = $FormatTables{$uid}{$formatId};
       
   351 			}
       
   352 		else
       
   353 			{
       
   354 			$formatString = $FormatTables{$category}{$subCategory};
       
   355 			}
       
   356 
       
   357 
       
   358 		# Get thread names
       
   359 		if ($category == $EThreadIdentification)
       
   360 			{
       
   361 			if ($subCategory == $EProcessName)
       
   362 				{
       
   363 				my $process = unpack("V", substr($recordData, 4, 4));
       
   364 				my $processName = substr($recordData, 8, $recordLen - 8);	
       
   365 #				printf ("\nprocess [%08X] processName [$processName]\n", $process);
       
   366 				$ProcessNames{$process} = $processName;
       
   367 				}
       
   368 			elsif ($subCategory == $EThreadCreate || $subCategory == $EThreadName)
       
   369 				{
       
   370 				my $thread = unpack("V", substr($recordData, 0, 4));
       
   371 				my $process = unpack("V", substr($recordData, 4, 4));
       
   372 				my $threadName = substr($recordData, 8, $recordLen - 8);	
       
   373 #				printf ("\nprocess [%08X] thread [%08X] threadName [$threadName]\n", $process, $thread, $threadName);
       
   374 				$ThreadNames{$thread} = $ProcessNames{$process} . "::" . $threadName;
       
   375 				}
       
   376 			elsif ($subCategory == $EThreadId)
       
   377 				{
       
   378 				my $thread = unpack("V", substr($recordData, 0, 4));
       
   379 				my $process = unpack("V", substr($recordData, 4, 4));
       
   380 				my $threadId = unpack("V", substr($recordData, 8, 4));
       
   381 #				printf ("\nprocess [%08X] thread [%08X] threadId [%08X]\n", $process, $thread, $threadId);
       
   382 				$ThreadIds{$thread} = $threadId;
       
   383 				}
       
   384 			}
       
   385 			
       
   386 			
       
   387 		# print Format string (col #9)
       
   388 		if ($PrintFlagFormatString)
       
   389 			{
       
   390 			my $formatStringWithoutCommas = $formatString;
       
   391 			$formatStringWithoutCommas=~ s/,/ /g;
       
   392 			printf "%s, ", $formatStringWithoutCommas;
       
   393 			}
       
   394 
       
   395 		my $formattedText;
       
   396 		
       
   397 		my $lenFormatString = length($formatString);
       
   398 		if ($lenFormatString && !$RawMode && $multipartFlags != $EMultipartMiddle && $multipartFlags != $EMultipartLast)
       
   399 			{
       
   400 			for (my $i=0; $i<$lenFormatString; $i++)
       
   401 				{
       
   402 				my $c = (substr ($formatString, $i, 1));
       
   403 #				printf "$c\n";
       
   404 				if ($c eq "%")
       
   405 					{
       
   406 					undef my $fieldLen;
       
   407 					$i++;
       
   408 	        		$c = (substr ($formatString, $i, 1));
       
   409 					if ($c eq "%")
       
   410 						{
       
   411 						$formattedText.= substr ($formatString, $i, 1);
       
   412 						next;
       
   413 						}
       
   414 					if ($c eq "*")	## take length from buffer
       
   415 						{
       
   416 						$fieldLen = unpack("V", substr($recordData, $recordPos, 4));
       
   417 						if ($fieldLen > $recordLen-$recordPos)
       
   418 							{
       
   419 							$formattedText.= "*** Invalid field length ***";
       
   420 							last;
       
   421 							}
       
   422 						$recordPos+= 4;
       
   423 						$i++;
       
   424 		        		$c = (substr ($formatString, $i, 1));
       
   425 						}
       
   426 					if (lc $c eq "x" || $c eq "h")
       
   427 						{
       
   428 						if (defined $fieldLen)
       
   429 							{
       
   430 							if (($fieldLen & 3) == 0)
       
   431 								{
       
   432 								for (my $i=0; $i< $fieldLen; $i+= 4)
       
   433 									{
       
   434 									$formattedText.= sprintf ("%08X ", unpack("V", substr($recordData, $recordPos, 4)));
       
   435 									$recordPos+= 4;
       
   436 									}
       
   437 								}
       
   438 							else
       
   439 								{
       
   440 								for (my $i=0; $i< $fieldLen; $i++)
       
   441 									{
       
   442 									$formattedText.= sprintf ("%02X ", unpack("C", substr($recordData, $recordPos, 1)));
       
   443 									$recordPos++;
       
   444 									}
       
   445 								}
       
   446 							}
       
   447 						else
       
   448 							{
       
   449 							$formattedText.= sprintf ("0x%08X", unpack("V", substr($recordData, $recordPos, 4)));
       
   450 							$recordPos+= 4;
       
   451 							}
       
   452 						$recordPos = ($recordPos + 3) & ~3;
       
   453 						next;
       
   454 						}
       
   455 					# display "%ld" as hex for now as don't know how to get perl to use or display a 64 decimal value
       
   456 					elsif (lc $c eq "l" && substr ($formatString, $i+1, 1) eq "d")
       
   457 						{
       
   458 						$i++;
       
   459 						my $loWord = unpack("V", substr($recordData, $recordPos, 4));
       
   460 						$recordPos+= 4;
       
   461 						my $hiWord = unpack("V", substr($recordData, $recordPos, 4));
       
   462 						$recordPos+= 4;
       
   463 						$formattedText.= sprintf ("0x%X:%08X", $hiWord, $loWord);
       
   464 						}
       
   465 					elsif (lc $c eq "l" && substr ($formatString, $i+1, 1) eq "x")
       
   466 						{
       
   467 						$i++;
       
   468 						my $loWord = unpack("V", substr($recordData, $recordPos, 4));
       
   469 						$recordPos+= 4;
       
   470 						my $hiWord = unpack("V", substr($recordData, $recordPos, 4));
       
   471 						$recordPos+= 4;
       
   472 						$formattedText.= sprintf ("0x%X:%08X", $hiWord, $loWord);
       
   473 						}
       
   474 					elsif (lc $c eq "d")
       
   475 						{
       
   476 						$formattedText.= sprintf ("%d", unpack("V", substr($recordData, $recordPos, 4)));
       
   477 						$recordPos+= 4;
       
   478 						$recordPos = ($recordPos + 3) & ~3;
       
   479 						next;
       
   480 						}
       
   481 					elsif ($c eq "s")
       
   482 						{
       
   483 						if (!defined $fieldLen) 
       
   484 							{$fieldLen = $recordLen - $recordPos;}
       
   485 						$formattedText.= substr($recordData, $recordPos, $fieldLen);
       
   486 						$recordPos+= $fieldLen; 
       
   487 						$recordPos = ($recordPos + 3) & ~3;
       
   488 						next;
       
   489 						}
       
   490 					elsif ($c eq "S")
       
   491 						{
       
   492 						if (!defined $fieldLen) 
       
   493 							{$fieldLen = $recordLen-$recordPos;}
       
   494 						for (my $j=0; $j < $fieldLen; $j+=2)
       
   495 							{
       
   496 					        my $byte = unpack("c", substr ($recordData, $recordPos+$j, 1));
       
   497  							$formattedText.= sprintf ("%c", $byte);
       
   498 							}
       
   499 						$recordPos+= $fieldLen; 
       
   500 						$recordPos = ($recordPos + 3) & ~3;
       
   501 						next;
       
   502 						}
       
   503 					elsif ($c eq "c")
       
   504 						{
       
   505 				        my $byte = unpack("c", substr ($recordData, $recordPos, 1));
       
   506 						$formattedText.= sprintf ("%c", $byte);
       
   507 						}
       
   508 					}
       
   509 				else
       
   510 					{
       
   511 					$formattedText.= $c;
       
   512 					}
       
   513 				}
       
   514 			}
       
   515 		else	# no format string : print as hex
       
   516 			{
       
   517 			for (my $i=0; $i < $recordLen; $i+=4)
       
   518 				{
       
   519 				$formattedText.= sprintf ("%08X ", unpack("V", substr($recordData, $i, 4)));
       
   520 				}
       
   521 			$recordPos+= $recordLen; $recordLen = 0;
       
   522 			
       
   523 			}
       
   524 		
       
   525 
       
   526 		# print Formatted text (col #10)
       
   527 		$formattedText=~ s/,/;/g;
       
   528 		$formattedText=~ s/\r//g;
       
   529 		$formattedText=~ s/\n/,/g;
       
   530 		printf "%s", $formattedText;
       
   531 
       
   532 		printf("\n");
       
   533 
       
   534 		if ($len < 0 || $recordLen < 0)	{die "truncated file";}
       
   535   
       
   536 
       
   537 		$pos+= ($len +3) & ~3;
       
   538 		seek (LOGFILE, $pos, SEEK_SET) or die "truncated file";
       
   539 		$i++;
       
   540 		}
       
   541 
       
   542 	close (LOGFILE);
       
   543 
       
   544 	if ($VerboseMode)
       
   545 		{
       
   546 		print "*** Processes ***\n";
       
   547 		for $id ( keys %ProcessNames )
       
   548 			{
       
   549 			printf ("process %08X ProcessName %s\n", $id, $ProcessNames{$id});
       
   550 			}
       
   551 		print "*** Thread ***\n";
       
   552 		for $id ( keys %ThreadNames )
       
   553 			{
       
   554 			printf ("thread %08X ThreadName %s::%X\n", $id, $ThreadNames{$id}, $ThreadIds{$id});
       
   555 			}
       
   556 		}
       
   557 
       
   558     }
       
   559 
       
   560     
       
   561 sub ReadSingleRecord
       
   562 	{
       
   563 	($fh, $data, $dataLen, $recordLen, $category, $subCategory, $multipartFlags, $extraN, $totalLen, $offset, $RawMode) = @_;	
       
   564 	
       
   565 	my $hdr;
       
   566 	my $flags;
       
   567 	my $header2;
       
   568 	my $timestamp;
       
   569 	my $timestamp2;
       
   570 	my $contextId;
       
   571 	my $programConter;	
       
   572 	
       
   573 	my $recordOffset = 0;
       
   574 	
       
   575 	$timestampLast;	
       
   576 	my $timestampDelta = 0;	
       
   577 	
       
   578 	my $bytesRead = read($fh, $hdr, 4);
       
   579 	
       
   580 	
       
   581 	if ($bytesRead < 4)	
       
   582 		{return -1;}
       
   583 
       
   584 	($$recordLen,$flags,$$category,$$subCategory) = unpack("CCCC", $hdr);
       
   585 	$$dataLen = $$recordLen-4;
       
   586 	
       
   587 	if ($flags & $EHeader2Present)
       
   588 		{$$multipartFlags = (ReadDword($fh) & $EMultipartFlagMask); $$dataLen-= 4}
       
   589 	else
       
   590 		{$$multipartFlags = 0;}
       
   591 	if ($flags & $ETimestampPresent)
       
   592 		{$timestamp = ReadDword($fh); $$dataLen-= 4;}
       
   593 	if ($flags & $ETimestamp2Present)
       
   594 		{$timestamp2 = ReadDword($fh); $$dataLen-= 4;}
       
   595 	if ($flags & $EContextIdPresent)
       
   596 		{$contextId = ReadDword($fh); $$dataLen-= 4;}
       
   597 	if ($flags & $EPcPresent)
       
   598 		{$programConter = ReadDword($fh); $$dataLen-= 4;}
       
   599 	if ($flags & $EExtraPresent)
       
   600 		{$$extraN = ReadDword($fh); $$dataLen-= 4;}
       
   601 	if ($$multipartFlags != 0)
       
   602 		{
       
   603 		$$totalLen = ReadDword($fh);  $$dataLen-= 4;
       
   604 		if ($$multipartFlags == $EMultipartMiddle || $$multipartFlags == $EMultipartLast)
       
   605 			{$$offset = ReadDword($fh);  $$totalLen-= 4; $$dataLen-= 4;}
       
   606 		}				
       
   607 
       
   608 	$timestampDelta = $timestamp - $timestampLast;
       
   609 	$timestampLast = $timestamp;
       
   610 
       
   611 	read($fh, $$data, ($$dataLen + 3) & ~3);
       
   612 
       
   613 
       
   614 	if ($RawMode || $$multipartFlags == $EMultipartFirst || $$multipartFlags == 0)
       
   615 		{
       
   616 		# print header len (col #1)
       
   617 		if ($PrintFlagHdrLen){printf ("0x%02X, ", $$recordLen);}
       
   618 	
       
   619 		# print header flags (col #2)
       
   620 		if ($PrintFlagHdrFlags)
       
   621 			{
       
   622 			printf ("%02X ", $flags);
       
   623 			if ($flags & $EHeader2Present) {printf "EHeader2Present ";}
       
   624 			if ($flags & $ETimestampPresent) {printf "ETimestampPresent ";}
       
   625 			if ($flags & $ETimestamp2Present) {printf "ETimestamp2Present ";}
       
   626 			if ($flags & $EContextIdPresent) {printf "EContextIdPresent ";}
       
   627 			if ($flags & $EPcPresent) {printf "EPcPresent ";}
       
   628 			if ($$multipartFlags != 0)
       
   629 				{
       
   630 				printf "EExtraPresent ";
       
   631 				if ($$multipartFlags == $EMultipartFirst) {print "EMultipartFirst ";}
       
   632 				elsif ($$multipartFlags == $EMultipartMiddle) {print "EMultipartMiddle ";}
       
   633 				elsif ($$multipartFlags == $EMultipartLast) {print "EMultipartLast ";}
       
   634 				printf ("ExtraN(0x%08X) ", $$extraN);
       
   635 				}
       
   636 			if ($flags & $ERecordTruncated) {printf "ERecordTruncated ";}
       
   637 			if ($flags & $EMissingRecord) {printf "EMissingRecord ";}
       
   638 			print ",";
       
   639 			}
       
   640 				
       
   641 		# print category (col #3)
       
   642 		printf "(%d;%d) $categoryString  , ", $$category, $$subCategory;
       
   643 	
       
   644 		# print timestamp(s) (col #4)
       
   645 		printf "0x";
       
   646 		if (defined $timestamp2) {printf "%08X : ", $timestamp2;}
       
   647 		printf "%08X", $timestamp;
       
   648 		printf ", ";;
       
   649 	
       
   650 		# print timestamp delta (col #5)
       
   651 		printf "0x%08X, ", $timestampDelta;
       
   652 
       
   653 		# print context Id (col #6)
       
   654 		if (!$RawMode && defined $ThreadNames{$contextId})
       
   655 			{
       
   656 			printf ("%s::%X, ", $ThreadNames{$contextId}, $ThreadIds{$contextId});
       
   657 			}
       
   658 		else			
       
   659 			{
       
   660 			if ((($contextId & $EContextIdMask) == $EContextIdThread) || $RawMode)
       
   661 				{printf "0x%08X, ", $contextId;}
       
   662 			elsif (($contextId & $EContextIdMask) == $EContextIdFIQ)
       
   663 				{printf "FIQ, ";}
       
   664 			elsif (($contextId & $EContextIdMask) == $EContextIdIRQ)
       
   665 				{printf "IRQ, ";}
       
   666 			elsif (($contextId & $EContextIdMask) == $EContextIdIDFC)
       
   667 				{printf "IDFC, ";}
       
   668 			}
       
   669 	
       
   670 		# print Program Counter (col #7)
       
   671 		printf "0x%08X, ", $programConter;
       
   672 		}
       
   673 
       
   674 		
       
   675 	
       
   676 	
       
   677 #########################################################
       
   678 #	my $hex;
       
   679 #	for (my $i=0; $i < $$dataLen; $i+=4)
       
   680 #		{
       
   681 #		$hex.= sprintf ("%08X ", unpack("V", substr($$data, $i, 4)));
       
   682 #		}
       
   683 #	printf "\nadding [$hex]\n";
       
   684 #########################################################
       
   685 	return $bytesRead
       
   686 	}
       
   687 
       
   688 	      
       
   689 sub ReadRecord 
       
   690 	{
       
   691 	($fh, $recordPos, $recordData, $category, $subCategory, $multipartFlags, $RawMode) = @_;
       
   692 #	printf "CurrentPos %08X\n", $pos;
       
   693 
       
   694 
       
   695 
       
   696 	seek ($fh, $$recordPos, SEEK_SET) or die "truncated file";
       
   697 	my $recordLen;
       
   698 	my $extraN;
       
   699 	my $totalLen;
       
   700 	my $offset;
       
   701 	my $dataLen;
       
   702 	my $data;
       
   703 	my $bytesRead;
       
   704 	
       
   705 	
       
   706 	$bytesRead = ReadSingleRecord($fh,  \$data, \$dataLen, \$recordLen, \$$category, \$$subCategory, \$$multipartFlags, \$extraN, \$totalLen, \$offset, $RawMode);
       
   707 
       
   708 	if ($bytesRead == -1)	# eof ?
       
   709 		{return -1; }
       
   710 	$$recordPos+= ($recordLen +3) & ~3;
       
   711 	
       
   712 	$$recordData = $data;
       
   713     $offset = $dataLen;
       
   714 
       
   715 	$offset-= 4;		# subtract 4 bytes for UID ?????????
       
   716     
       
   717     if ($RawMode || $$multipartFlags != $EMultipartFirst)
       
   718     	{return $dataLen;}
       
   719 
       
   720     $pos = $$recordPos;
       
   721 
       
   722 	while (1)
       
   723 		{
       
   724 		
       
   725 		# find next record, i.e. look for a record which matches $extraN 
       
   726 		
       
   727 		seek ($fh, $pos, SEEK_SET) or die "truncated file";
       
   728 
       
   729 		my $recordLen;
       
   730 		
       
   731 		my $category;
       
   732 		my $subCategory;
       
   733 		my $multipartFlags;
       
   734 		my $currentExtraN;
       
   735 		my $currentOffset;
       
   736 		
       
   737 		my $totalLen;
       
   738 		my $currentDataLen;
       
   739 		my $data;
       
   740 		$bytesRead = ReadSingleRecord($fh, \$data, \$currentDataLen, \$recordLen, \$category, \$subCategory, \$multipartFlags, \$currentExtraN, \$totalLen, \$currentOffset, $RawMode);
       
   741 		if ($bytesRead == -1)	# eof ?
       
   742 			{return -1; }
       
   743 		$pos+= ($recordLen +3) & ~3;
       
   744 		
       
   745 #		printf "\npos %08X, Seaching for (extra %08X, offset %08X), found (extra %08X, offset %08X)\n",
       
   746 #			$pos, $extraN, $offset, $currentExtraN, $currentOffset;
       
   747 
       
   748 		if ($currentExtraN == $extraN && $currentOffset == $offset)
       
   749 			{
       
   750 			$$recordData.= $data;
       
   751 			$offset+= $currentDataLen;
       
   752 			$dataLen+= $currentDataLen;
       
   753 			}
       
   754 			
       
   755 		if ($multipartFlags == $EMultipartLast)
       
   756 			{last;}
       
   757 		}
       
   758 	
       
   759 	return $dataLen;
       
   760 	}	
       
   761 
       
   762 sub ReadDword {
       
   763 	(my $fh) = @_;
       
   764 	my $buffer;
       
   765 
       
   766 	$bytesRead = read($fh, $buffer, 4);
       
   767 	if ($bytesRead < 4) 	{die "truncated file";}
       
   768 
       
   769 	my $dword = unpack("V", $buffer);
       
   770 
       
   771 	return $dword
       
   772 	};
       
   773 
       
   774 sub ReadByte {
       
   775 	(my $fh) = @_;
       
   776 	my $buffer;
       
   777 
       
   778 	$bytesRead = read($fh, $buffer, 1);
       
   779 	if ($bytesRead < 1) 	{die "truncated file";}
       
   780 
       
   781 	my $byte = unpack("C", $buffer);
       
   782 
       
   783 	return $byte
       
   784 	};
       
   785 
       
   786     
       
   787 	
       
   788 sub PrintFormatTables($)
       
   789 	{
       
   790 	my ($formatTables) = @_;
       
   791 		
       
   792 	for $tableIndex ( sort keys %$formatTables )
       
   793 		{
       
   794 		printf ("SYMTraceFormatCategory %08X:\n", $tableIndex);
       
   795 		for $formatId (sort keys %{ $$formatTables{$tableIndex} } )
       
   796 			{
       
   797 			printf ("%08X => %s\n", $formatId, $$formatTables{$tableIndex}{$formatId});
       
   798 			}
       
   799 			print "\n";
       
   800 		}
       
   801 	}
       
   802         
       
   803 
       
   804 
       
   805 sub OutputSawDictionary($)
       
   806 	{
       
   807 	my ($formatTables) = @_;
       
   808 
       
   809 
       
   810 	# SAW enums
       
   811 	$EFieldTypeHexDump = 0;
       
   812 	$EFieldTypeHex = 1;
       
   813 	$EFieldTypeDecimal = 2;
       
   814 	$EFieldTypeStringToEnd = 3;
       
   815 	$EFieldTypeNullTerminatedString = 4;
       
   816 	$EFieldTypeHexDumpToEnd = 5;
       
   817 	$EFieldTypeUnicodeToEnd = 6;
       
   818 	$EFieldTypeNullTerminatedUnicode = 7;
       
   819 	$EFieldTypeCountedUnicode = 8;
       
   820 	$EFieldTypeCountedHexDump = 9;
       
   821 	$EFieldTypeCountedString = 10;
       
   822 
       
   823 	my $moduleIds;	# string containg all UIDs separared by semi-colons
       
   824 		
       
   825 	for $tableIndex ( sort keys %$formatTables )
       
   826 		{
       
   827 		if ($tableIndex < 256)
       
   828 			{
       
   829 			next;
       
   830 			}
       
   831 		$moduleIds.= sprintf ("%08X;", $tableIndex);
       
   832 		
       
   833 		printf ("MODULEID_%08X_DESC=\n", $tableIndex);
       
   834 		printf ("MODULEID_%08X_NAME=%08X\n", $tableIndex, $tableIndex);
       
   835 		
       
   836 		my $formatIds;
       
   837 		$formatIds = sprintf ("MODULEID_%08X_FORMATIDS=", $tableIndex);
       
   838 		
       
   839 		for $formatId  (sort keys %{ $$formatTables{$tableIndex} } )
       
   840 			{
       
   841 			$formatIds.= sprintf ("%d;", $formatId);
       
   842 			}
       
   843 		printf ("$formatIds\n");
       
   844 		
       
   845 		
       
   846 		for $formatId (sort keys %{ $$formatTables{$tableIndex} } )
       
   847 			{
       
   848 			my $fieldCount = 0;
       
   849 			my $formatString = $$formatTables{$tableIndex}{$formatId};
       
   850 			
       
   851 #printf ("formatString = (%s)\n", $formatString);
       
   852 
       
   853 			# format name is the first format string up until the first space or '%' character or end-of line ...
       
   854 			$formatString=~ m/^[^%\s]*/;
       
   855 			my $formatName = $&;
       
   856 			
       
   857 			# thow the format name away
       
   858 			$formatString = $';
       
   859 			
       
   860 			# strip the leading space
       
   861 			$formatString=~ s/\s*//;
       
   862 
       
   863 			printf ("MODULEID_%08X_FORMATID_%d_NAME=%s\n", $tableIndex, $formatId, $formatName);
       
   864 #printf ("MODULEID_%08X_FORMATID_%d_DESC=\n", $tableIndex, $formatId);
       
   865 
       
   866 			my $lenFormatString = length($formatString);
       
   867 			
       
   868 			my $formattedText;
       
   869 			my $fieldType = $EFieldTypeHex;
       
   870 			my $fieldLen = 0;
       
   871 			while (length($formatString))
       
   872 				{
       
   873 				my $c = (substr ($formatString, 0, 1));
       
   874 #print ("[$formatString][$c]\n");				
       
   875 				$formatString=~ s/.//;	# strip the leading space
       
   876 				if ($c eq "%")
       
   877 					{
       
   878 #print "found %\n";							
       
   879 					my $fieldLenSpecified = 0;
       
   880 	        		$c = (substr ($formatString, 0, 1));
       
   881 					$formatString=~ s/.//;	# discard char
       
   882 #print "c2=$c\n";							
       
   883 					if ($c eq "%")
       
   884 						{
       
   885 						$formattedText.= substr ($formatString, 0, 1);
       
   886 						next;
       
   887 						}
       
   888 					if ($c eq "*")	## take length from buffer
       
   889 						{
       
   890 						$fieldLenSpecified = 1;
       
   891 		        		$c = (substr ($formatString, 0, 1));
       
   892 						$formatString=~ s/.//;	# discard char
       
   893 						}
       
   894 					if (lc $c eq "x" || $c eq "h")
       
   895 						{
       
   896 						## deal wilth $fieldLenSpecified
       
   897 						if ($fieldLenSpecified)
       
   898 							{
       
   899 							$fieldType = $EFieldTypeCountedHexDump;
       
   900 							$fieldLen = 0;
       
   901 							}
       
   902 						else
       
   903 							{
       
   904 							$fieldType = $EFieldTypeHex;
       
   905 							$fieldLen = 4;
       
   906 							}
       
   907 						}
       
   908 					elsif (lc $c eq "l" && substr ($formatString, 0, 1) eq "d")
       
   909 						{
       
   910 						$formatString=~ s/.//;	# discard char
       
   911 						$fieldType = $EFieldTypeDecimal;
       
   912 						$fieldLen = 8;
       
   913 						}
       
   914 					elsif (lc $c eq "l" && substr ($formatString, 0, 1) eq "x")
       
   915 						{
       
   916 						$formatString=~ s/.//;	# discard char
       
   917 						$fieldType = $EFieldTypeHex;
       
   918 						$fieldLen = 8;
       
   919 						}
       
   920 					elsif (lc $c eq "d")
       
   921 						{
       
   922 						$fieldType = $EFieldTypeDecimal;
       
   923 						$fieldLen = 4;
       
   924 						}
       
   925 					elsif ($c eq "s")
       
   926 						{
       
   927 						## deal wilth $fieldLenSpecified
       
   928 						if ($fieldLenSpecified)
       
   929 							{
       
   930 							$fieldType = $EFieldTypeCountedString;
       
   931 							$fieldLen = 0;
       
   932 							}
       
   933 						else
       
   934 							{
       
   935 							$fieldType = $EFieldTypeStringToEnd;
       
   936 							$fieldLen = 0;
       
   937 							}
       
   938 						}
       
   939 					elsif ($c eq "S")
       
   940 						{
       
   941 						## deal wilth $fieldLenSpecified
       
   942 						if ($fieldLenSpecified)
       
   943 							{
       
   944 							$fieldType = $EFieldTypeCountedUnicode;
       
   945 							$fieldLen = 0;
       
   946 							}
       
   947 						else
       
   948 							{
       
   949 							$fieldType = EFieldTypeUnicodeToEnd;
       
   950 							$fieldLen = 0;
       
   951 							}
       
   952 						}
       
   953 					elsif ($c eq "c")
       
   954 						{
       
   955 						$fieldType = $EFieldTypeHex;
       
   956 						$fieldLen = 1;
       
   957 						}
       
   958 					printf ("MODULEID_%08X_FORMATID_%d_FIELD_%d_NAME=%s\n", $tableIndex, $formatId, $fieldCount, $formattedText);
       
   959 					printf ("MODULEID_%08X_FORMATID_%d_FIELD_%d_TYPE=%s\n", $tableIndex, $formatId, $fieldCount, $fieldType);
       
   960 					if ($fieldLen > 0)
       
   961 						{printf ("MODULEID_%08X_FORMATID_%d_FIELD_%d_LENGTH=%s\n", $tableIndex, $formatId, $fieldCount, $fieldLen);}
       
   962 					$fieldCount++;
       
   963 					$formattedText="";
       
   964 					
       
   965 					$formatString=~ s/\s//;	# strip the leading space
       
   966 					}
       
   967 				else
       
   968 					{
       
   969 #					if ($c eq ":") {$formattedText.= '\\'; }
       
   970 					$formattedText.= $c;
       
   971 					}
       
   972 				}
       
   973 			printf ("MODULEID_%08X_FORMATID_%d_FIELDS=%d\n", $tableIndex, $formatId, $fieldCount);
       
   974 			
       
   975 			}
       
   976 		print "MODULEIDS=$moduleIds\n";
       
   977 		}
       
   978 	}
       
   979 	
       
   980 	
       
   981 	
       
   982 	
       
   983 	
       
   984 	
       
   985 	        
       
   986         
       
   987 sub H2Trace($$)
       
   988 {
       
   989 	%basictypes = (
       
   990 		TInt8		=>	1,
       
   991 		TUint8		=>	1,
       
   992 		TInt16		=>	2,
       
   993 		TUint16		=>	2,
       
   994 		TInt32		=>	4,
       
   995 		TUint32		=>	4,
       
   996 		TInt		=>	4,
       
   997 		TUint		=>	4,
       
   998 		TBool		=>	4,
       
   999 		TInt64		=>	8,
       
  1000 		TUint64		=>	8,
       
  1001 		TLinAddr	=>	4,
       
  1002 		TVersion	=>	4,
       
  1003 		TPde		=>	4,
       
  1004 		TPte		=>	4,
       
  1005 		TProcessPriority => 4,
       
  1006 		TFormatId	=>  2,
       
  1007 	);
       
  1008 	
       
  1009 	if (scalar(@_)!= 2) {
       
  1010 		die "perl h2trace.pl <input.h>\n";
       
  1011 	}
       
  1012 	my ($infile, $filescope) = @_;
       
  1013 	
       
  1014 	if ($VerboseMode)
       
  1015 		{print "\nOpening $infile\n";}
       
  1016 	
       
  1017 	open IN, $infile or die "Can't open $infile for input\n";
       
  1018 	my $in;
       
  1019 	while (<IN>) {
       
  1020 		$in.=$_;
       
  1021 	}
       
  1022 	close IN;
       
  1023 	
       
  1024 	# First remove any backslash-newline combinations
       
  1025 	$in =~ s/\\\n//gms;
       
  1026 	
       
  1027 	# Remove any character constants
       
  1028 	$in =~  s/\'(.?(${0})*?)\'//gms;
       
  1029 	
       
  1030 	# Strip comments beginning with //
       
  1031 	$in =~ s/\/\/(.*?)\n/\n/gms;    #//(.*?)\n
       
  1032 	
       
  1033 	# Strip comments (/* */) but leave doxygen comments (/** */)
       
  1034 	$in =~ s/\/\*[^*](.*?)\*\//\n/gms;  #/*(.*?)*/
       
  1035 	
       
  1036 	
       
  1037 	# Collapse whitespace into a single space or newline
       
  1038 	$in =~ s/\t/\ /gms;
       
  1039 	$in =~ s/\r/\ /gms;
       
  1040 	
       
  1041 	# Tokenize on non-identifier characters
       
  1042 	my @tokens0 = split(/(\W)/,$in);
       
  1043 	my @tokens;
       
  1044 	my $inString = 0;
       
  1045 	my $inComment = 0;
       
  1046 	my $string;
       
  1047 	foreach $t (@tokens0) {
       
  1048 		next if ($t eq "");
       
  1049 		next if (!$inString && ($t eq " " or $t eq ""));
       
  1050 		if ($inComment == 0) 
       
  1051 			{
       
  1052 			if ($t eq "/")
       
  1053 				{$inComment = 1;}
       
  1054 			}
       
  1055 		elsif ($inComment == 1) 
       
  1056 			{
       
  1057 			if ($t eq "*")
       
  1058 				{$inComment = 2;}
       
  1059 			else
       
  1060 				{$inComment = 0;}
       
  1061 			}
       
  1062 		elsif ($inComment == 2) 
       
  1063 			{
       
  1064 			if ($t eq "*")
       
  1065 				{$inComment = 3;}
       
  1066 			}
       
  1067 		elsif ($inComment == 3) 
       
  1068 			{
       
  1069 			if ($t eq "/")
       
  1070 				{
       
  1071 				$inComment = 0;
       
  1072 		        # if we were in a string, need to push previous '*'
       
  1073 		        if ($inString)
       
  1074 		          {
       
  1075 		          push @tokens, "*";
       
  1076 		          }
       
  1077 				$inString = 0;	# end of comment aborts a string
       
  1078 				$string = "";
       
  1079 				}
       
  1080 			else
       
  1081 				{$inComment = 2;}
       
  1082 			}
       
  1083 			
       
  1084 		if ($t eq "\"")
       
  1085 			{
       
  1086 			if (!$inString) 
       
  1087 				{
       
  1088 				$inString=1;
       
  1089 				next;
       
  1090 				}
       
  1091 			else
       
  1092 				{
       
  1093 				$inString=0;
       
  1094 				$t = $string;
       
  1095 				$string = "";
       
  1096 #				if ($VerboseMode) {print "string : [$t]\n";	}
       
  1097 				}
       
  1098 			}
       
  1099 			
       
  1100 		if ($inString)
       
  1101 			{
       
  1102 			$string.= $t;
       
  1103 			next;
       
  1104 			}
       
  1105 		push @tokens, $t;
       
  1106 	}
       
  1107 	
       
  1108 	my $CurrentTraceFormatString;
       
  1109 	my $CurrentTraceFormatCategory;
       
  1110 	# format Key as specified by the @TraceFormatCategory tag is either the current category 
       
  1111 	# or the current UID
       
  1112 	my $CurrentFormatTableKey;	
       
  1113 	
       
  1114 	
       
  1115 	my $line=1;
       
  1116 	parse_scope($filescope, \@tokens, \$line);
       
  1117 
       
  1118 	#print $in;
       
  1119 	#print join (" ", @tokens);
       
  1120 }	# end of     H2Trace
       
  1121 	
       
  1122 
       
  1123 
       
  1124 	sub parse_scope($$$) {
       
  1125 		my ($scope, $tokens, $line) = @_;
       
  1126 		my $state = 1;
       
  1127 		
       
  1128 		my @classes;
       
  1129 		my $curr_offset=0;
       
  1130 		my $overall_align=0;
       
  1131 #		print ">parse_scope $scope->{name}\n";
       
  1132 		
       
  1133 		while (scalar(@$tokens))
       
  1134 			{
       
  1135 			my $t = shift @$tokens;
       
  1136 #			printf "t: [$t] [$$line]\n";
       
  1137 	    	if (!defined ($t)) {
       
  1138 	      		printf "undefined !";
       
  1139 	      		next;
       
  1140 	      	}
       
  1141 			if ($state>=-1 and $t eq "\n") {
       
  1142 				++$$line;
       
  1143 				$state=1;
       
  1144 				next;
       
  1145 			} elsif ($state==-1 and $t ne "\n") {
       
  1146 				next;
       
  1147 			} elsif ($state==-2 and $t ne ';') {
       
  1148 				next;
       
  1149 			}
       
  1150 			
       
  1151 			if ($state>0 and $t eq '#') {
       
  1152 				$t = shift @$tokens;
       
  1153 				if ($t eq 'define') {
       
  1154 					my $ident = shift @$tokens;
       
  1155 					my $defn = shift @$tokens;
       
  1156 					if ($defn ne '(') {	# don't do macros with parameters
       
  1157 #					print "MACRO: $ident :== $defn\n";
       
  1158 					$macros{$ident} = $defn;
       
  1159 					}
       
  1160 				}
       
  1161 				$state=-1;	# skip to next line
       
  1162 				next;
       
  1163 			}
       
  1164 			
       
  1165 			
       
  1166 			if (parse_doxygen($scope,$tokens, $line, $t) == 1)
       
  1167 				{next;}
       
  1168 	
       
  1169 			if ($t eq "namespace" ) {
       
  1170 				$state=0;
       
  1171 				my %cl;
       
  1172 				$cl{specifier}=$t;
       
  1173 				$cl{scope}=$scope;
       
  1174 				$cl{values}=$scope->{values};
       
  1175 				$cl{members}=\$scope->{members};
       
  1176 				$cl{typedefs}=\$scope->{typedefs};
       
  1177 				$cl{FormatTables}=$scope->{FormatTables};
       
  1178 				$cl{formatStrings} =$scope->{formatStrings};
       
  1179 				$cl{formatCategories} =$scope->{formatCategories};
       
  1180 				
       
  1181 				my $new_namespace = \%cl;
       
  1182 				my $n = get_token($scope,$tokens,$line);
       
  1183 				if ($n !~ /\w+/) {
       
  1184 					warn "Unnamed $t not supported at line $$line\n";
       
  1185 					return;
       
  1186 				}
       
  1187 				$new_namespace->{name}=$n;
       
  1188 				my @class_match = grep {$_->{name} eq $n} @classes;
       
  1189 				my $exists = scalar(@class_match);
       
  1190 				my $b = get_token($scope,$tokens,$line);
       
  1191 				if ($b eq ':') {
       
  1192 					die "Inheritance not supported at line $$line\n";
       
  1193 				} elsif ($b eq ';') {
       
  1194 					# forward declaration
       
  1195 					push @classes, $new_namespace unless ($exists);
       
  1196 					next;
       
  1197 				} elsif ($b ne '{') {
       
  1198 					warn "Syntax error#1 at line $$line\n";
       
  1199 					return;
       
  1200 				}
       
  1201 				if ($exists) {
       
  1202 					$new_namespace = $class_match[0];
       
  1203 					if ($new_namespace->{complete}) {
       
  1204 						warn "Duplicate definition of $cl{specifier} $n\n";
       
  1205 					}
       
  1206 				}
       
  1207 				push @classes, $new_namespace unless ($exists);
       
  1208 				parse_scope($new_namespace, $tokens, $line);
       
  1209 				next;
       
  1210 			}
       
  1211 			
       
  1212 			if ($t eq "struct" or $t eq "class" or $t eq "NONSHARABLE_CLASS") {
       
  1213 				next if ($state==0);
       
  1214 				$state=0;
       
  1215 				my %cl;
       
  1216 				$cl{specifier}=$t;
       
  1217 				$cl{scope}=$scope;
       
  1218 				my @members;
       
  1219 				my @typedefs;
       
  1220 				$cl{members}=\@members;
       
  1221 				$cl{typedefs}=\@typedefs;
       
  1222 				$cl{FormatTables}=$scope->{FormatTables};
       
  1223 				my $new_class = \%cl;
       
  1224 				my $n;
       
  1225 
       
  1226 				if ($t eq "NONSHARABLE_CLASS")
       
  1227 					{
       
  1228 					my $b = get_token($scope,$tokens,$line);
       
  1229 					if ($b !~ /\(/) {die "Syntax error at line $$line\n";}
       
  1230 					$n = get_token($scope,$tokens,$line);
       
  1231   				$b = get_token($scope,$tokens,$line);
       
  1232 					if ($b !~ /\)/) {die "Syntax error at line $$line\n";}
       
  1233 					}
       
  1234 				else					
       
  1235 					{
       
  1236 					$n = get_token($scope,$tokens,$line);
       
  1237 					}
       
  1238 								
       
  1239 				
       
  1240 				if ($n !~ /\w+/) {
       
  1241 					warn "Unnamed $t not supported at line $$line\n";
       
  1242 					return;
       
  1243 				}
       
  1244 				$new_class->{name}=$n;
       
  1245 				my @class_match = grep {$_->{name} eq $n} @classes;
       
  1246 				my $exists = scalar(@class_match);
       
  1247 				my $b = get_token($scope,$tokens,$line);
       
  1248 				#skip inheritance etc until we get to a '{' or \ ';'
       
  1249 				while ($b ne '{' && $b ne ';')
       
  1250 					{
       
  1251 			        $b = get_token($scope,$tokens,$line);
       
  1252 			        die "Syntax error#2 at line $$line\n" if  (!defined $b);
       
  1253 					}
       
  1254 				if ($b eq ';') {
       
  1255 					# forward declaration
       
  1256 					push @classes, $new_class unless ($exists);
       
  1257 					next;
       
  1258 				} 
       
  1259 				if ($exists) {
       
  1260 					$new_class = $class_match[0];
       
  1261 					if ($new_class->{complete}) {
       
  1262 						warn "Duplicate definition of $cl{specifier} $n\n";
       
  1263 					}
       
  1264 				}
       
  1265 				push @classes, $new_class unless ($exists);
       
  1266 				parse_scope($new_class, $tokens, $line);
       
  1267 				next;
       
  1268 			} elsif ($t eq "enum") {
       
  1269 				$state=0;
       
  1270 				my $n = get_token($scope,$tokens,$line);
       
  1271 				my $name="";
       
  1272 				if ($n =~ /\w+/) {
       
  1273 					$name = $n;
       
  1274 					$n = get_token($scope,$tokens,$line);
       
  1275 				}
       
  1276 				push @enums, $name;
       
  1277 				if ($n ne '{') {
       
  1278 					die "Syntax error#4 at line $$line\n";
       
  1279 				}
       
  1280 				parse_enum($scope, $tokens, $line, $name);
       
  1281 				next;
       
  1282 			} elsif ($t eq '}') {
       
  1283 				$state=0;
       
  1284 				if ($scope->{scope}) {
       
  1285 			        if ($scope->{specifier} eq "namespace")
       
  1286 			        	{
       
  1287 						$scope->{complete}=1;
       
  1288 #						print "Scope completed\n";
       
  1289 						last;
       
  1290 						}
       
  1291 					$t = get_token($scope,$tokens,$line);
       
  1292 					# skip to next ';'
       
  1293 					while (defined ($t) and $t ne ';')
       
  1294 						{$t = get_token($scope,$tokens,$line);}
       
  1295 					die "Syntax error#5 at line $$line\n" if ($t ne ';');
       
  1296 					$scope->{complete}=1;
       
  1297 #					print "Scope completed\n";
       
  1298 					last;
       
  1299 				}
       
  1300 				warn "Syntax error#5 at line $$line\n";
       
  1301 				return;
       
  1302 			}
       
  1303 			$state=0;
       
  1304 			if ($scope->{scope}) {
       
  1305 				if ($t eq "public" or $t eq "private" or $t eq "protected") {
       
  1306 					if (shift (@$tokens) eq ':') {
       
  1307 						next;	# ignore access specifiers
       
  1308 					}
       
  1309 				die "Syntax error#6 at line $$line\n";
       
  1310 				}
       
  1311 			}
       
  1312 			unshift @$tokens, $t;
       
  1313 			
       
  1314 			my @currdecl = parse_decl_def($scope, $tokens, $line);
       
  1315 #			print scalar (@currdecl), "\n";
       
  1316 			if ($t eq 'static') {
       
  1317 				next;	# skip static members
       
  1318 			}
       
  1319 			my $typedef;
       
  1320 			if ($t eq 'typedef') {
       
  1321 #			print "TYPEDEF\n";
       
  1322 				$typedef = 1;
       
  1323 				$t = shift @currdecl;
       
  1324 				$t = $currdecl[0];
       
  1325 			} else {
       
  1326 #			print "NOT TYPEDEF\n";
       
  1327 				$typedef = 0;
       
  1328 			}
       
  1329 #			print "$currdecl[0]\n";
       
  1330 			next if (scalar(@currdecl)==0);
       
  1331 			
       
  1332 			if ($t eq "const") {
       
  1333 				# check for constant declaration
       
  1334 #				print "CONST $currdecl[1] $currdecl[2] $currdecl[3]\n";
       
  1335 				my $ctype = lookup_type($scope, $currdecl[1]);
       
  1336 #				print "$ctype->{basic}    $ctype->{size}\n";
       
  1337 				if ($ctype->{basic} and $currdecl[2]=~/^\w+$/ and $currdecl[3] eq '=') {
       
  1338 					if ($typedef!=0) {
       
  1339 						die "Syntax error#7 at line $$line\n";
       
  1340 					}
       
  1341 					shift @currdecl;
       
  1342 					shift @currdecl;
       
  1343 					my $type = $ctype->{name};
       
  1344 					my $name;		#### = shift @currdecl;
       
  1345 
       
  1346 					if ($scope->{name})
       
  1347 						{	
       
  1348 						$name = $scope->{name} . "::" . shift @currdecl;
       
  1349 						}
       
  1350 					else
       
  1351 						{
       
  1352 						$name = shift @currdecl;
       
  1353 						}
       
  1354 #					printf "[$name,$scope->{name}]";
       
  1355 					my $size = $ctype->{size};
       
  1356 					shift @currdecl;
       
  1357 					my $value = get_constant_expr($scope,\@currdecl,$line);
       
  1358 					$values{$name} = {type=>$type, size=>$size, value=>$value};
       
  1359 					next;
       
  1360 				}
       
  1361 			}
       
  1362 			
       
  1363 			
       
  1364 			
       
  1365 		}
       
  1366 	}
       
  1367 	
       
  1368 	sub get_token($$$) {
       
  1369 		my ($scope,$tokenlist,$line) = @_;
       
  1370 		while (scalar(@$tokenlist)) {
       
  1371 			my $t = shift @$tokenlist;
       
  1372 			return $t if (!defined($t));
       
  1373 			if (parse_doxygen($scope,$tokenlist, $line, $t) == 1)
       
  1374 				{next;}
       
  1375 			if ($t !~ /^[\s]*$/)
       
  1376 				{
       
  1377 				if ($$tokenlist[0] eq ":" and $$tokenlist[1] eq ":")
       
  1378 					{
       
  1379 					$t.= shift @$tokenlist;
       
  1380 					$t.= shift @$tokenlist;
       
  1381 					$t.= shift @$tokenlist;
       
  1382 #					print "Colon-separated token";
       
  1383 					}
       
  1384 				return $t
       
  1385 				}
       
  1386 			++$$line;
       
  1387 		}
       
  1388   		return undef;
       
  1389 	}
       
  1390 	
       
  1391 	sub skip_qualifiers($) {
       
  1392 		my ($tokens) = @_;
       
  1393 		my $f=0;
       
  1394 		my %quals = (
       
  1395 			EXPORT_C => 1,
       
  1396 			IMPORT_C => 1,
       
  1397 			inline => 1,
       
  1398 			virtual => 0,
       
  1399 			const => 0,
       
  1400 			volatile => 0,
       
  1401 			static => 0,
       
  1402 			extern => 0,
       
  1403 			LOCAL_C => 0,
       
  1404 			LOCAL_D => 0,
       
  1405 			GLDEF_C => 0,
       
  1406 			GLREF_C => 0,
       
  1407 			GLDEF_D => 0,
       
  1408 			GLREF_D => 0
       
  1409 			);
       
  1410 		for (;;) {
       
  1411 			my $t = $$tokens[0];
       
  1412 			my $q = $quals{$t};
       
  1413 			last unless (defined ($q));
       
  1414 			$f |= $q;
       
  1415 			shift @$tokens;
       
  1416 		}
       
  1417 		return $f;
       
  1418 	}
       
  1419 	
       
  1420 	sub parse_indirection($) {
       
  1421 		my ($tokens) = @_;
       
  1422 		my $level = 0;
       
  1423 		for (;;) {
       
  1424 			my $t = $$tokens[0];
       
  1425 			if ($t eq '*') {
       
  1426 				++$level;
       
  1427 				shift @$tokens;
       
  1428 				next;
       
  1429 			}
       
  1430 			last if ($t ne "const" and $t ne "volatile");
       
  1431 			shift @$tokens;
       
  1432 		}
       
  1433 		return $level;
       
  1434 	}
       
  1435 	
       
  1436 	sub get_operand($$$) {
       
  1437 		my ($scope,$tokens,$line) = @_;
       
  1438 		my $t = get_token($scope,$tokens,$line);
       
  1439 		if ($t eq '-') {
       
  1440 			my $x = get_operand($scope,$tokens,$line);
       
  1441 			return -$x;
       
  1442 		} elsif ($t eq '+') {
       
  1443 			my $x = get_operand($scope,$tokens,$line);
       
  1444 			return $x;
       
  1445 		} elsif ($t eq '~') {
       
  1446 			my $x = get_operand($scope,$tokens,$line);
       
  1447 			return ~$x;
       
  1448 		} elsif ($t eq '!') {
       
  1449 			my $x = get_operand($scope,$tokens,$line);
       
  1450 			return $x ? 0 : 1;
       
  1451 		} elsif ($t eq '(') {
       
  1452 			my $x = get_constant_expr($scope,$tokens,$line);
       
  1453 			my $t = get_token($scope,$tokens,$line);
       
  1454 			if ($t ne ')') {
       
  1455 				warn "Missing ) at line $$line\n";
       
  1456 				return undefined;
       
  1457 			}
       
  1458 			return $x;
       
  1459 		} elsif ($t eq "sizeof") {
       
  1460 			my $ident = get_token($scope,$tokens,$line);
       
  1461 			if ($ident eq '(') {
       
  1462 				$ident = get_token($scope,$tokens,$line);
       
  1463 				my $cb = get_token($scope,$tokens,$line);
       
  1464 				if ($cb ne ')') {
       
  1465 					warn "Bad sizeof() syntax at line $$line\n";
       
  1466 					return undefined;
       
  1467 				}
       
  1468 			}
       
  1469 			$ident = look_through_macros($ident);
       
  1470 			if ($ident !~ /^\w+$/) {
       
  1471 				warn "Bad sizeof() syntax at line $$line\n";
       
  1472 				return undefined;
       
  1473 			}
       
  1474 			my $type = lookup_type($scope, $ident);
       
  1475 			if (!defined $type) {
       
  1476 				warn "Unrecognised type $ident at line $$line\n";
       
  1477 				return undefined;
       
  1478 			}
       
  1479 			if ($type->{basic}) {
       
  1480 				return $type->{size};
       
  1481 			} elsif ($type->{enum}) {
       
  1482 				return 4;
       
  1483 			} elsif ($type->{ptr}) {
       
  1484 				return 4;
       
  1485 			} elsif ($type->{fptr}) {
       
  1486 				return 4;
       
  1487 			}
       
  1488 			my $al = $type->{class}->{align};
       
  1489 			my $sz = $type->{class}->{size};
       
  1490 			return ($sz+$al-1)&~($al-1);
       
  1491 		}
       
  1492 		$t = look_through_macros($t);
       
  1493 		if ($t =~ /^0x/i) {
       
  1494 			return oct($t);
       
  1495 		} elsif ($t =~ /^\d/) {
       
  1496 			return $t;
       
  1497 		} elsif ($t =~ /^\w+$/) {
       
  1498 			my $x = lookup_value($scope,$t);
       
  1499 #			die "Unrecognised identifier '$t' at line $$line\n" unless defined($x);
       
  1500 			if (!defined($x)) {
       
  1501 				print "Unrecognised identifier '$t' at line $$line\n" ;
       
  1502 			}
       
  1503 			return $x;
       
  1504 		} elsif ($t =~ /^\w+::\w+$/) {
       
  1505 			my $x = lookup_value($scope,$t);
       
  1506 #			die "Unrecognised identifier '$t' at line $$line\n" unless defined($x);
       
  1507 			if (!defined($x)) {
       
  1508 				print "Unrecognised identifier '$t' at line $$line\n" ;
       
  1509 			}
       
  1510 			return $x;
       
  1511 		} else {
       
  1512 			warn "Syntax error#10 at line $$line\n";
       
  1513 			return undefined;
       
  1514 		}
       
  1515 	}
       
  1516 	
       
  1517 	sub look_through_macros($) {
       
  1518 		my ($ident) = @_;
       
  1519 		while ($ident and $macros{$ident}) {
       
  1520 			$ident = $macros{$ident};
       
  1521 		}
       
  1522 		return $ident;
       
  1523 	}
       
  1524 	
       
  1525 	sub lookup_value($$) {
       
  1526 		my ($scope,$ident) = @_;
       
  1527 		while ($scope) {
       
  1528 			my $vl = $scope->{values};
       
  1529 			if (defined($vl->{$ident})) {
       
  1530 				return $vl->{$ident}->{value};
       
  1531 			}
       
  1532 			$scope = $scope->{scope};
       
  1533 		}
       
  1534 		return undef();
       
  1535 	}
       
  1536 	
       
  1537 	sub lookup_type($$) {
       
  1538 		my ($scope,$ident) = @_;
       
  1539 		if ($basictypes{$ident}) {
       
  1540 			return {scope=>$scope, basic=>1, name=>$ident, size=>$basictypes{$ident} };
       
  1541 		}
       
  1542 		while ($scope) {
       
  1543 			if ($basictypes{$ident}) {
       
  1544 				return {scope=>$scope, basic=>1, name=>$ident, size=>$basictypes{$ident} };
       
  1545 			}
       
  1546 			my $el = $scope->{enums};
       
  1547 			my $cl = $scope->{classes};
       
  1548 			my $td = $scope->{typedefs};
       
  1549 			if (grep {$_ eq $ident} @$el) {
       
  1550 				return {scope=>$scope, enum=>1, name=>$ident, size=>4 };
       
  1551 			}
       
  1552 			my @match_class = (grep {$_->{name} eq $ident} @$cl);
       
  1553 			if (scalar(@match_class)) {
       
  1554 				return {scope=>$scope, class=>$match_class[0]};
       
  1555 			}
       
  1556 			my @match_td = (grep {$_->{name} eq $ident} @$td);
       
  1557 			if (scalar(@match_td)) {
       
  1558 				my $tdr = $match_td[0];
       
  1559 				my $cat = $tdr->{category};
       
  1560 				if ($cat eq 'basic' or $cat eq 'enum' or $cat eq 'class') {
       
  1561 					$ident = $tdr->{alias};
       
  1562 					next;
       
  1563 				} else {
       
  1564 					return { scope=>$scope, $cat=>1, $size=>$tdr->{size} };
       
  1565 				}
       
  1566 			}
       
  1567 			$scope = $scope->{scope};
       
  1568 		}
       
  1569 		return undef();
       
  1570 	}
       
  1571 	
       
  1572 	sub get_mult_expr($$$) {
       
  1573 		my ($scope,$tokens,$line) = @_;
       
  1574 		my $x = get_operand($scope,$tokens,$line);
       
  1575 		my $t;
       
  1576 		for (;;) {
       
  1577 			$t = get_token($scope,$tokens,$line);
       
  1578 			if ($t eq '*') {
       
  1579 				my $y = get_operand($scope,$tokens,$line);
       
  1580 				$x = $x * $y;
       
  1581 			} elsif ($t eq '/') {
       
  1582 				my $y = get_operand($scope,$tokens,$line);
       
  1583 				if ($y != 0)
       
  1584 					{$x = int($x / $y);}
       
  1585 			} elsif ($t eq '%') {
       
  1586 				my $y = get_operand($scope,$tokens,$line);
       
  1587 				if ($y != 0)
       
  1588 					{$x = int($x % $y);}
       
  1589 			} else {
       
  1590 				last;
       
  1591 			}
       
  1592 		}
       
  1593 		unshift @$tokens, $t;
       
  1594 		return $x;
       
  1595 	}
       
  1596 	
       
  1597 	sub get_add_expr($$$) {
       
  1598 		my ($scope,$tokens,$line) = @_;
       
  1599 		my $x = get_mult_expr($scope,$tokens,$line);
       
  1600 		my $t;
       
  1601 		for (;;) {
       
  1602 			$t = get_token($scope,$tokens,$line);
       
  1603 			if ($t eq '+') {
       
  1604 				my $y = get_mult_expr($scope,$tokens,$line);
       
  1605 				$x = $x + $y;
       
  1606 			} elsif ($t eq '-') {
       
  1607 				my $y = get_mult_expr($scope,$tokens,$line);
       
  1608 				$x = $x - $y;
       
  1609 			} else {
       
  1610 				last;
       
  1611 			}
       
  1612 		}
       
  1613 		unshift @$tokens, $t;
       
  1614 		return $x;
       
  1615 	}
       
  1616 	
       
  1617 	sub get_shift_expr($$$) {
       
  1618 		my ($scope,$tokens,$line) = @_;
       
  1619 		my $x = get_add_expr($scope,$tokens,$line);
       
  1620 		my $t, $t2;
       
  1621 		for (;;) {
       
  1622 			$t = get_token($scope,$tokens,$line);
       
  1623 			if ($t eq '<' or $t eq '>') {
       
  1624 				$t2 = get_token($scope,$tokens,$line);
       
  1625 				if ($t2 ne $t) {
       
  1626 					unshift @$tokens, $t2;
       
  1627 					last;
       
  1628 				}
       
  1629 			}
       
  1630 			if ($t eq '<') {
       
  1631 				my $y = get_add_expr($scope,$tokens,$line);
       
  1632 				$x = $x << $y;
       
  1633 			} elsif ($t eq '>') {
       
  1634 				my $y = get_add_expr($scope,$tokens,$line);
       
  1635 				$x = $x >> $y;
       
  1636 			} else {
       
  1637 				last;
       
  1638 			}
       
  1639 		}
       
  1640 		unshift @$tokens, $t;
       
  1641 		return $x;
       
  1642 	}
       
  1643 	
       
  1644 	sub get_and_expr($$$) {
       
  1645 		my ($scope,$tokens,$line) = @_;
       
  1646 		my $x = get_shift_expr($scope,$tokens,$line);
       
  1647 		my $t;
       
  1648 		for (;;) {
       
  1649 			$t = get_token($scope,$tokens,$line);
       
  1650 			if ($t eq '&') {
       
  1651 				my $y = get_shift_expr($scope,$tokens,$line);
       
  1652 				$x = $x & $y;
       
  1653 			} else {
       
  1654 				last;
       
  1655 			}
       
  1656 		}
       
  1657 		unshift @$tokens, $t;
       
  1658 		return $x;
       
  1659 	}
       
  1660 	
       
  1661 	sub get_xor_expr($$$) {
       
  1662 		my ($scope,$tokens,$line) = @_;
       
  1663 		my $x = get_and_expr($scope,$tokens,$line);
       
  1664 		my $t;
       
  1665 		for (;;) {
       
  1666 			$t = get_token($scope,$tokens,$line);
       
  1667 			if ($t eq '^') {
       
  1668 				my $y = get_and_expr($scope,$tokens,$line);
       
  1669 				$x = $x ^ $y;
       
  1670 			} else {
       
  1671 				last;
       
  1672 			}
       
  1673 		}
       
  1674 		unshift @$tokens, $t;
       
  1675 		return $x;
       
  1676 	}
       
  1677 	
       
  1678 	sub get_ior_expr($$$) {
       
  1679 		my ($scope,$tokens,$line) = @_;
       
  1680 		my $x = get_xor_expr($scope,$tokens,$line);
       
  1681 		my $t;
       
  1682 		for (;;) {
       
  1683 			$t = get_token($scope,$tokens,$line);
       
  1684 			if ($t eq '|') {
       
  1685 				my $y = get_xor_expr($scope,$tokens,$line);
       
  1686 				$x = $x | $y;
       
  1687 			} else {
       
  1688 				last;
       
  1689 			}
       
  1690 		}
       
  1691 		unshift @$tokens, $t;
       
  1692 		return $x;
       
  1693 	}
       
  1694 	
       
  1695 	sub get_constant_expr($$$) {
       
  1696 		my ($scope,$tokens,$line) = @_;
       
  1697 		my $x = get_ior_expr($scope,$tokens,$line);
       
  1698 		return $x;
       
  1699 	}
       
  1700 	
       
  1701 	sub parse_enum($$$$) {
       
  1702 		my ($scope,$tokens,$line,$enum_name) = @_;
       
  1703 		my $vl = $scope->{values};
       
  1704 		my $fstr = $scope->{formatStrings};
       
  1705 		my $fcat = $scope->{formatCategories};
       
  1706 		my $fmtTable = $scope->{FormatTables};
       
  1707 		
       
  1708 		my $x = 0;
       
  1709 		for (;;) {
       
  1710 			my $t = get_token($scope,$tokens,$line);
       
  1711 			last if ($t eq '}');
       
  1712 			if (!defined($t)) {
       
  1713 				die "Unexpected end of file #2 at line $$line\n";
       
  1714 			}
       
  1715 			
       
  1716 			if ($t eq '#') {
       
  1717 				next;
       
  1718 				}
       
  1719 			
       
  1720 			if ($t !~ /^\w+$/) {
       
  1721 				warn "Syntax error#11 at line $$line\n";
       
  1722 				next;
       
  1723 			}
       
  1724 
       
  1725 			if ($scope->{name})
       
  1726 				{	
       
  1727 				$t = $scope->{name} . "::" . $t;
       
  1728 				}
       
  1729 
       
  1730 			if (defined($vl->{$t})) {
       
  1731 				warn "Duplicate identifier [$t] at line $$line\n";
       
  1732 			}
       
  1733 			my $t2 = get_token($scope,$tokens,$line);
       
  1734 			if ($t2 eq ',') {
       
  1735 				$vl->{$t} = {type=>$enum_name, size=>4, value=>$x, enum=>1};
       
  1736 				$fstr->{$t} = $CurrentTraceFormatString; 
       
  1737 				$fcat->{$t} = $CurrentTraceFormatCategory; 
       
  1738 				if (defined $CurrentTraceFormatCategory && defined $CurrentTraceFormatString)
       
  1739 					{ $fmtTable->{$CurrentTraceFormatCategory}{$x} = $CurrentTraceFormatString; }
       
  1740 				undef $CurrentTraceFormatString;
       
  1741 				++$x;
       
  1742 			} elsif ($t2 eq '}') {
       
  1743 				$vl->{$t} = {type=>$enum_name, size=>4, value=>$x, enum=>1};
       
  1744 				$fstr->{$t} = $CurrentTraceFormatString; 
       
  1745 				$fcat->{$t} = $CurrentTraceFormatCategory; 
       
  1746 				if (defined $CurrentTraceFormatCategory && defined $CurrentTraceFormatString)
       
  1747 					{ $fmtTable->{$CurrentTraceFormatCategory}{$x} = $CurrentTraceFormatString; }
       
  1748 				undef $CurrentTraceFormatString;
       
  1749 				++$x;
       
  1750 				last;
       
  1751 			} elsif ($t2 eq '=') {
       
  1752 				$x = get_constant_expr($scope, $tokens, $line);
       
  1753 				$vl->{$t} = {type=>$enum_name, size=>4, value=>$x, enum=>1};
       
  1754 				$fstr->{$t} = $CurrentTraceFormatString; 
       
  1755 				$fcat->{$t} = $CurrentTraceFormatCategory;
       
  1756 				if (defined $CurrentTraceFormatCategory && defined $CurrentTraceFormatString)
       
  1757 					{ $fmtTable->{$CurrentTraceFormatCategory}{$x} = $CurrentTraceFormatString; }
       
  1758 				undef $CurrentTraceFormatString; 
       
  1759 				++$x;
       
  1760 				$t2 = get_token($scope,$tokens,$line);
       
  1761 				last if ($t2 eq '}');
       
  1762 				next if ($t2 eq ',');
       
  1763 				warn "Syntax error#12 at line $$line\n";
       
  1764 			} else {
       
  1765 				unshift @$tokens, $t2;
       
  1766 			}
       
  1767 		}
       
  1768 		my $t = get_token($scope,$tokens,$line);
       
  1769 		if ($t ne ';') {
       
  1770 			warn "Missing ; at line $$line\n";
       
  1771 		}
       
  1772 	}
       
  1773 	
       
  1774 	
       
  1775 	sub  parse_decl_def($$$) {
       
  1776 		my ($scope,$tokens,$line) = @_;
       
  1777 		my $level=0;
       
  1778 		my @decl;
       
  1779 		while ( scalar(@$tokens) ) {
       
  1780 			my $t = get_token($scope,$tokens, $line);
       
  1781 			if ( (!defined ($t) || $t eq ';') and ($level==0)) {
       
  1782 				return @decl;
       
  1783 			}
       
  1784 	
       
  1785 			if ($t eq "static")
       
  1786 				{
       
  1787 				next;
       
  1788 				}
       
  1789 	
       
  1790 			push @decl, $t;
       
  1791 			if ($t eq '{') {
       
  1792 				++$level;
       
  1793 			}
       
  1794 			if ($t eq '}') {
       
  1795 				if ($level==0) {
       
  1796 					warn "Syntax error#13 at line $$line\n";
       
  1797 					unshift @$tokens, $t;
       
  1798 					return @decl;
       
  1799 					
       
  1800 				}
       
  1801 				if (--$level==0) {
       
  1802 					return ();	# end of function definition reached
       
  1803 				}
       
  1804 			}
       
  1805 		}
       
  1806 		die "Unexpected end of file #3 at line $$line\n";
       
  1807 	}
       
  1808 	
       
  1809 	sub dump_scope($) {
       
  1810 		my ($scope) = @_;
       
  1811 		my $el = $scope->{enums};
       
  1812 		my $cl = $scope->{classes};
       
  1813 		my $vl = $scope->{values};
       
  1814 		my $fstr = $scope->{formatStrings};
       
  1815 		my $fcat = $scope->{formatCategories};
       
  1816 		print "SCOPE: $scope->{name}\n";
       
  1817 		if (scalar(@$el)) {
       
  1818 			print "\tenums:\n";
       
  1819 			foreach (@$el) {
       
  1820 				print "\t\t$_\n";
       
  1821 			}
       
  1822 		}
       
  1823 		if (scalar(keys(%$vl))) {
       
  1824 			print "\tvalues:\n";
       
  1825 			foreach $vname (keys(%$vl)) {
       
  1826 				my $v = $vl->{$vname};
       
  1827 				my $x = $v->{value};
       
  1828 				my $t = $v->{type};
       
  1829 				my $sz = $v->{size};
       
  1830 				my $fstring = $fstr->{$vname};
       
  1831 				my $fcategory = $fcat->{$vname};
       
  1832 				if ($v->{enum}) {
       
  1833 					printf ("\t\t$vname\=$x (enum $t) size=$sz fcat=[0x%x] fstr=[%s]\n", $fcategory,$fstring);
       
  1834 				} else {
       
  1835 					printf ("\t\t$vname\=$x (type $t) size=$sz fcat=[0x%x] fstr=[%s]\n", $fcategory, $fstring);
       
  1836 				}
       
  1837 			}
       
  1838 		}
       
  1839 		if ($scope->{scope}) {
       
  1840 			my $members = $scope->{members};
       
  1841 			foreach (@$members) {
       
  1842 				my $n = $_->{name};
       
  1843 				my $sz = $_->{size};
       
  1844 				my $off = $_->{offset};
       
  1845 				my $spc = $_->{spacing};
       
  1846 				if (defined $spc) {
       
  1847 					print "\t$n\[\]\: spacing $spc size $sz offset $off\n";
       
  1848 				} else {
       
  1849 					print "\t$n\: size $sz offset $off\n";
       
  1850 				}
       
  1851 			}
       
  1852 			print "\tOverall size : $scope->{size}\n";
       
  1853 			print "\tOverall align: $scope->{align}\n";
       
  1854 		}
       
  1855 		foreach $s (@$cl) {
       
  1856 			dump_scope($s);
       
  1857 		}
       
  1858 	}
       
  1859 	
       
  1860 	
       
  1861 	
       
  1862 		
       
  1863 	sub parse_doxygen($$$$) {
       
  1864 		my ($scope,$tokens,$line,$t) = @_;
       
  1865 	
       
  1866 		if ($t ne "/")
       
  1867 			{
       
  1868 			return 0;	# not a doxygen comment
       
  1869 			}
       
  1870 		if ($t eq "/") {
       
  1871 			$state=0;
       
  1872 			my $t2 = shift @$tokens;
       
  1873 			my $t3 = shift @$tokens;
       
  1874 	
       
  1875 			if ($t2 ne "*" || $t3 ne "*")
       
  1876 				{
       
  1877 				unshift @$tokens, $t3;
       
  1878 				unshift @$tokens, $t2;
       
  1879 				return 0;	# not a doxygen comment
       
  1880 				}
       
  1881 		}
       
  1882 #		printf "doxygen start on line %d\n", $$line;
       
  1883 		for (;;) {
       
  1884 			my $t = shift @$tokens;
       
  1885 			if (!defined($t)) 
       
  1886 					{
       
  1887 					warn "Unexpected end of file #4 at line $$line\n";	
       
  1888 					return
       
  1889 					}
       
  1890 			
       
  1891 			if ($t eq "\n"){++$$line };
       
  1892 			
       
  1893 			if ($t eq '*')
       
  1894 				{
       
  1895 				my $t2 = shift @$tokens;
       
  1896 				last if ($t2 eq '/');
       
  1897 				unshift @$tokens, $t2;
       
  1898 				}
       
  1899 			
       
  1900 			if ($t eq '@')
       
  1901 				{
       
  1902 				my $t2 = shift @$tokens;
       
  1903 				if ($t2 eq 'SYMTraceFormatString')
       
  1904 					{
       
  1905 					my $t3 = shift @$tokens;
       
  1906 #					if ($VerboseMode){print "SYMTraceFormatString = [$t3]\n";}
       
  1907 					$CurrentTraceFormatString = $t3;
       
  1908 					}
       
  1909 				if ($t2 eq 'SYMTraceFormatCategory')
       
  1910 					{
       
  1911 					$CurrentTraceFormatCategory = get_operand($scope,$tokens,$line);
       
  1912 #					if ($VerboseMode){printf ("SYMTraceFormatCategory = 0x%x\n", $CurrentTraceFormatCategory);}
       
  1913 					}
       
  1914 				else
       
  1915 					{
       
  1916 					unshift @$tokens, $t2;
       
  1917 					}
       
  1918 				}
       
  1919 	
       
  1920 		}
       
  1921 #		printf ("doxygen end  on line %d\n", $$line);
       
  1922 		return 1;	# is a doxygen comment
       
  1923 	}
       
  1924 	
       
  1925 
       
  1926         
       
  1927         
       
  1928         
       
  1929         
       
  1930         
       
  1931         
       
  1932         
       
  1933         
       
  1934         
       
  1935         
       
  1936         
       
  1937         
       
  1938         
       
  1939         
       
  1940         
       
  1941         
       
  1942         
       
  1943         
       
  1944         
       
  1945         
       
  1946         
       
  1947         
       
  1948         
       
  1949         
       
  1950         
       
  1951         
       
  1952