kerneltest/e32utils/trace/btrace_analyse.cpp
changeset 0 a41df078684a
child 43 c1f20ce4abcf
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/kerneltest/e32utils/trace/btrace_analyse.cpp	Mon Oct 19 15:55:17 2009 +0100
@@ -0,0 +1,3110 @@
+// Copyright (c) 2007-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:
+//
+
+#if defined(__EPOC32__) || defined(__WINS__)
+// compiling for Symbian OS...
+#include <e32std.h>
+#include <e32std_private.h>
+#include <e32svr.h>
+#include <e32def.h>
+#include <e32def_private.h>
+#include <d32btrace.h>
+#include "e32btrace.h"
+
+inline void* malloc(TInt aSize) { return User::Alloc(aSize); }
+inline void* realloc(void* aPtr,TInt aSize) { return User::ReAlloc(aPtr,aSize); }
+inline void free(void* aPtr) { User::Free(aPtr); }
+
+TUint8 PrintfBuffer[256];
+TUint PrintfBufferWidth = 0;
+
+static void printf(const char* aFormat,...)
+	{
+	VA_LIST list;
+	VA_START(list,aFormat);
+	TPtrC8 format((const TUint8*)aFormat);
+	TPtr8 buf(PrintfBuffer+PrintfBufferWidth,sizeof(PrintfBuffer)-PrintfBufferWidth);
+	buf.AppendFormatList(format,list);
+	PrintfBufferWidth += buf.Size();
+	for(;;)
+		{
+		TUint width = 0;
+		for(;;)
+			{
+			if(width>=PrintfBufferWidth)
+				return;
+			if(PrintfBuffer[width]=='\n')
+				break;
+			++width;
+			}
+		TPtrC8 line(PrintfBuffer,width);
+		++width;
+		RDebug::RawPrint(line);
+		_LIT(KLineEnd,"\r\n");
+		RDebug::RawPrint(KLineEnd);
+		memcpy(PrintfBuffer,PrintfBuffer+width,PrintfBufferWidth-width);
+		PrintfBufferWidth -= width;
+		}
+	}
+
+typedef TUint uintptr_t;
+
+#ifndef ASSERT
+#define ASSERT(c) (void)((c)||(AssertFailed(__LINE__)))
+TInt AssertFailed(TInt aLine)
+	{
+	_LIT(KPanicCategory,"BTRACE-ANALYSE");
+	User::Panic(KPanicCategory,aLine);
+	return 0;
+	}
+#endif // ASSERT
+
+
+//
+//	Trace buffer helper functions - for use on target only
+//
+RBTrace Trace;
+TUint8* AnalyseData;
+TUint AnalyseDataRemain = 0;
+
+void RePrime()
+	{
+	for(TInt i=0; i<256; ++i)
+		{
+		if(Trace.Filter(i))
+			{
+			// toggle filter to force a prime
+			Trace.SetFilter(i,0);
+			Trace.SetFilter(i,1);
+			}
+		}
+	}
+
+TUint AnalyseStream(TAny* aBuffer, TUint aMaxSize)
+	{
+	TUint size = AnalyseDataRemain;
+	if(!size)
+		{
+		Trace.DataUsed();
+		AnalyseDataRemain = Trace.GetData(AnalyseData);
+		size = AnalyseDataRemain;
+		}
+	if(size>aMaxSize)
+		size = aMaxSize;
+	memcpy(aBuffer,AnalyseData,size);
+	AnalyseData += size;
+	AnalyseDataRemain -= size;
+	return size;
+	}
+
+void ProcessAllTrace(TUint (*aInput)(TAny* aBuffer, TUint aMaxSize),TInt aReportLevel);
+
+void DoAnalyse(TInt aAnalysisLevel)
+	{
+	AnalyseDataRemain = 0;
+	TUint oldMode = Trace.Mode();
+	Trace.SetMode(0); // turn off trace capture while we dump
+	ProcessAllTrace(AnalyseStream, aAnalysisLevel);
+	Trace.SetMode(oldMode);
+	RePrime();
+	}
+
+TInt BTraceAnalyseSetup()
+	{
+	TInt r = Trace.Open();
+	if (r != KErrNone)
+		{
+		return r;
+		}
+	// Stop tracing
+	TUint oldMode = Trace.Mode();
+	Trace.SetMode(0);
+
+	// empty btrace buffer and reprime it
+	Trace.Empty();
+	Trace.SetMode(oldMode);
+	RePrime();
+	return KErrNone;
+	}
+
+void BTraceAnalyseEnd()
+	{
+	Trace.Close();
+	}
+
+void BTraceAnalyse(TInt aAnalysisLevel)
+	{
+	// Stop tracing
+	TUint oldMode = Trace.Mode();
+	Trace.SetMode(0);
+
+	AnalyseDataRemain = 0;
+	ProcessAllTrace(AnalyseStream, aAnalysisLevel);
+
+	// empty btrace buffer and reprime it
+	Trace.Empty();
+	Trace.SetMode(oldMode);
+	RePrime();
+	}
+
+#else
+// compiling for host...
+
+#include <stdio.h>
+#include <string.h>
+#include <malloc.h>
+
+#if defined(_MSC_VER)
+typedef __int64 longlong;
+typedef unsigned __int64 ulonglong;
+#define BREAKPOINT { _asm int 3 }			/**< Invoke debugger */
+#else
+typedef long long longlong;
+typedef long long ulonglong;
+#define BREAKPOINT
+#endif
+
+typedef signed char			TInt8;
+typedef unsigned char		TUint8;
+typedef unsigned short		TUint16;
+typedef unsigned int		TUint32;
+typedef ulonglong			TUint64;
+typedef unsigned int		uintptr_t;
+typedef int					TInt;
+typedef unsigned			TUint;
+typedef int					TBool;
+typedef void				TAny;
+typedef float				TReal;
+#define IMPORT_C
+#include "e32btrace.h"
+
+#define ASSERT(c) (void)((c)||(AssertFailed(__LINE__)))
+extern "C" void exit(int);
+TInt AssertFailed(TInt aLine)
+	{
+	fprintf(stderr,"Panic: BTRACE-ANALYSE %d",aLine);
+	BREAKPOINT;
+	exit(2);
+	return 0;
+	}
+
+#define __UHEAP_MARK
+#define __UHEAP_MARKEND
+
+TAny* operator new(size_t, TAny* p) {return p;}
+
+#endif // SYMBIAN_OS
+
+
+
+//
+// Global data
+//
+
+TInt ReportLevel = 0;
+TUint8 TraceBuffer[0x10000];
+TUint TraceBufferSize = 0;
+TUint64	Timestamp = 0;
+TUint64	TimestampBase = 0;
+TUint32	TraceRecordId = 0;
+TUint32	TimestampPeriod = 0;
+TUint32	Timestamp2Period = 0;
+TBool Timestamp64Bit = 0;
+TUint TraceFormatErrors = 0;
+TBool TraceBufferFilled = false;
+TBool SMP = true;
+TBool ErrorOnThisTrace = false;
+
+const TUint8 KJunkTraceSubcategory = 255; // Dummy sub-category for EMetaTrace
+
+const TInt KMaxCpus = 8;
+
+//
+// Utilities
+//
+
+void ToHex(TUint8* aString,TUint32 aValue)
+	{
+	TUint i=32;
+	do
+		{
+		i -= 4;
+		TUint8 c = (TUint8)((aValue>>i)&0xf);
+		if(c>=10)
+			c += 'a'-10;
+		else
+			c += '0';
+		*aString++ = c;
+		}
+	while(i);
+	}
+
+
+TUint MakeName(TUint8* aString,const char* aName,TUint32 aHexValue)
+	{
+	TUint8* start = aString;
+	for(;;)
+		{
+		TUint8 c = *aName++;
+		if(!c)
+			break;
+		*aString++ = c;
+		}
+	ToHex(aString,aHexValue);
+	aString[8] = 0;
+	return aString-start+8;
+	}
+
+
+/**
+Convert a timestamp into real time units (micro-seconds)
+*/
+TUint32 Time(TUint64 aTimestamp)
+	{
+	if(!TimestampPeriod)
+		return (TUint32)aTimestamp;
+	TInt exponent = (TInt8)(TimestampPeriod>>24);
+	TUint64 mantissa = TimestampPeriod&0xffffff;
+	aTimestamp *= mantissa;
+	exponent += 32;
+	if(exponent<0)
+		aTimestamp >>= -exponent;
+	else
+		aTimestamp <<= exponent;
+	TUint64 timeLo = (aTimestamp&0xffffffffu)*1000000;
+	TUint64 timeHi = (aTimestamp>>32)*1000000;
+	TUint64 us = timeHi+(timeLo>>32)+((timeLo>>31)&1);
+	return TUint32(us);
+	}
+
+
+void ReportTimeUnits()
+	{
+	if(!TimestampPeriod)
+		printf("\nAll times are given in BTrace Timestamp1 units.\n\n");
+	else
+		{
+		TInt exponent = (TInt8)(TimestampPeriod>>24);
+		TUint64 mantissa = TimestampPeriod&0xffffff;
+		TUint64 period = 1000000;
+		period *= mantissa;
+		exponent += 32;
+		if(exponent<0)
+			period >>= -exponent;
+		else
+			period <<= exponent;
+		printf("\nAll times are given in microseconds, resolution %d.%03dus\n\n",(int)TUint32(period>>32),(int)TUint32((((period&0xffffffffu)*1000)+0x80000000u)>>32));
+		}
+	}
+
+
+TUint SetBits(TUint8* aData, TUint aOffset, TUint aSize)
+	{
+	TUint mask = 1<<(aOffset&7);
+	TUint8* data = aData+(aOffset>>3);
+	TUint errors = 0;
+	while(aSize)
+		{
+		if(*data&mask)
+			++errors;
+		*data |= (TUint8)mask;
+		mask <<= 1;
+		if(mask>0xFF)
+			{
+			mask = 0x01;
+			++data;
+			}
+		--aSize;
+		}
+	return errors;
+	}
+
+
+TUint ClearBits(TUint8* aData, TUint aOffset, TUint aSize)
+	{
+	TUint mask = 1<<(aOffset&7);
+	TUint8* data = aData+(aOffset>>3);
+	TUint errors = 0;
+	while(aSize)
+		{
+		if(!(*data&mask))
+			++errors;
+		*data &= (TUint8)~mask;
+		mask <<= 1;
+		if(mask>0xFF)
+			{
+			mask = 0x01;
+			++data;
+			}
+		--aSize;
+		}
+	return errors;
+	}
+
+
+void WarnIfError(TUint aErrorCount)
+	{
+	if (TraceBufferFilled)
+		printf("WARNING - BTRACE BUFFER IS FULL SO TRACE DATA MAY BE INCOMPLETE\n");
+
+	if(aErrorCount==0 && TraceFormatErrors==0)
+		return;
+	printf("CONSISTENCY ERRORS FOUND DURING TRACE ANALYSIS, RESULTS ARE UNRELIABLE!\n");
+	}
+
+
+#define CHECK_TRACE_DATA_WORDS(numWords)	\
+	if(aTrace.iDataSize<numWords*4 && ++TraceFormatErrors) return
+
+
+//
+// Category naming
+//
+
+const char* const UnknownNames[256] =
+	{
+	"?00","?01","?02","?03","?04","?05","?06","?07","?08","?09",
+	"?10","?11","?12","?13","?14","?15","?16","?17","?18","?19",
+	"?20","?21","?22","?23","?24","?25","?26","?27","?28","?29",
+	"?30","?31","?32","?33","?34","?35","?36","?37","?38","?39",
+	"?40","?41","?42","?43","?44","?45","?46","?47","?48","?49",
+	"?50","?51","?52","?53","?54","?55","?56","?57","?58","?59",
+	"?60","?61","?62","?63","?64","?65","?66","?67","?68","?69",
+	"?70","?71","?72","?73","?74","?75","?76","?77","?78","?79",
+	"?80","?81","?82","?83","?84","?85","?86","?87","?88","?89",
+	"?90","?91","?92","?93","?94","?95","?96","?97","?98","?99",
+	"?100","?101","?102","?103","?104","?105","?106","?107","?108","?109",
+	"?110","?111","?112","?113","?114","?115","?116","?117","?118","?119",
+	"?120","?121","?122","?123","?124","?125","?126","?127","?128","?129",
+	"?130","?131","?132","?133","?134","?135","?136","?137","?138","?139",
+	"?140","?141","?142","?143","?144","?145","?146","?147","?148","?149",
+	"?150","?151","?152","?153","?154","?155","?156","?157","?158","?159",
+	"?160","?161","?162","?163","?164","?165","?166","?167","?168","?169",
+	"?170","?171","?172","?173","?174","?175","?176","?177","?178","?179",
+	"?180","?181","?182","?183","?184","?185","?186","?187","?188","?189",
+	"?190","?191","?192","?193","?194","?195","?196","?197","?198","?199",
+	"?200","?201","?202","?203","?204","?205","?206","?207","?208","?209",
+	"?210","?211","?212","?213","?214","?215","?216","?217","?218","?219",
+	"?220","?221","?222","?223","?224","?225","?226","?227","?228","?229",
+	"?230","?231","?232","?233","?234","?235","?236","?237","?238","?239",
+	"?240","?241","?242","?243","?244","?245","?246","?247","?248","?249",
+	"?250","?251","?252","?253","?254","?255"
+	};
+
+
+#define STRINGIFY2(a) #a						/**< Helper for #STRINGIFY */
+#define STRINGIFY(a) STRINGIFY2(a)				/**< Convert \a a into a quoted string */
+#define CASE_CAT_NAME(name) case BTrace::name: return STRINGIFY(name)
+
+const char* CategoryName(TUint8 aCategory)
+	{
+	switch((BTrace::TCategory)aCategory)
+		{
+		CASE_CAT_NAME(ERDebugPrintf);
+		CASE_CAT_NAME(EKernPrintf);
+		CASE_CAT_NAME(EPlatsecPrintf);
+		case BTrace::EThreadIdentification: return "EThreadId"; //	CASE_CAT_NAME(EThreadIdentification)
+		CASE_CAT_NAME(ECpuUsage);
+    	CASE_CAT_NAME(EKernPerfLog);
+    	CASE_CAT_NAME(EClientServer);
+    	CASE_CAT_NAME(ERequests);
+    	CASE_CAT_NAME(EChunks);
+    	CASE_CAT_NAME(ECodeSegs);
+		CASE_CAT_NAME(EPaging);
+		CASE_CAT_NAME(EThreadPriority);
+		CASE_CAT_NAME(EPagingMedia);
+		CASE_CAT_NAME(EKernelMemory);
+		CASE_CAT_NAME(EHeap);
+		CASE_CAT_NAME(EMetaTrace);
+
+		CASE_CAT_NAME(EFastMutex);
+		CASE_CAT_NAME(EProfiling);
+		CASE_CAT_NAME(ESymbianKernelSync);
+		CASE_CAT_NAME(EFlexibleMemModel);
+		CASE_CAT_NAME(ETest1);
+		CASE_CAT_NAME(ETest2);
+		default:
+			break;
+		}
+	return UnknownNames[aCategory];
+	}
+
+const char* SubCategoryName(TUint8 aCategory, TUint8 aSubCategory)
+	{
+	switch(aCategory)
+		{
+	case BTrace::ERDebugPrintf:
+	case BTrace::EKernPrintf:
+	case BTrace::EPlatsecPrintf:
+		return ""; // no subcategories for these
+
+	case BTrace::EThreadIdentification:
+		switch((BTrace::TThreadIdentification)aSubCategory)
+			{
+		CASE_CAT_NAME(ENanoThreadCreate);
+		CASE_CAT_NAME(ENanoThreadDestroy);
+		CASE_CAT_NAME(EThreadCreate);
+		CASE_CAT_NAME(EThreadDestroy);
+		CASE_CAT_NAME(EThreadName);
+		CASE_CAT_NAME(EProcessName);
+		CASE_CAT_NAME(EThreadId);
+		CASE_CAT_NAME(EProcessCreate);
+		CASE_CAT_NAME(EProcessDestroy);
+			}
+		break;
+
+	case BTrace::ECpuUsage:
+		switch((BTrace::TCpuUsage)aSubCategory)
+			{
+		CASE_CAT_NAME(EIrqStart);
+		CASE_CAT_NAME(EIrqEnd);
+		CASE_CAT_NAME(EFiqStart);
+		CASE_CAT_NAME(EFiqEnd);
+		CASE_CAT_NAME(EIDFCStart);
+		CASE_CAT_NAME(EIDFCEnd);
+		CASE_CAT_NAME(ENewThreadContext);
+			}
+		break;
+
+	case BTrace::EChunks:
+		switch((BTrace::TChunks)aSubCategory)
+			{
+		CASE_CAT_NAME(EChunkCreated);
+		CASE_CAT_NAME(EChunkInfo);
+		CASE_CAT_NAME(EChunkDestroyed);
+		CASE_CAT_NAME(EChunkMemoryAllocated);
+		CASE_CAT_NAME(EChunkMemoryDeallocated);
+		CASE_CAT_NAME(EChunkMemoryAdded);
+		CASE_CAT_NAME(EChunkMemoryRemoved);
+		CASE_CAT_NAME(EChunkOwner);
+			}
+		break;
+
+	case BTrace::ECodeSegs:
+		switch((BTrace::TCodeSegs)aSubCategory)
+			{
+		CASE_CAT_NAME(ECodeSegCreated);
+		CASE_CAT_NAME(ECodeSegInfo);
+		CASE_CAT_NAME(ECodeSegDestroyed);
+		CASE_CAT_NAME(ECodeSegMapped);
+		CASE_CAT_NAME(ECodeSegUnmapped);
+		CASE_CAT_NAME(ECodeSegMemoryAllocated);
+		CASE_CAT_NAME(ECodeSegMemoryDeallocated);
+			}
+		break;
+		
+	case BTrace::EPaging:
+		switch((BTrace::TPaging)aSubCategory)
+			{
+		CASE_CAT_NAME(EPagingPageInBegin);
+		CASE_CAT_NAME(EPagingPageInUnneeded);
+		CASE_CAT_NAME(EPagingPageInROM);
+		CASE_CAT_NAME(EPagingPageOutROM);
+		CASE_CAT_NAME(EPagingPageInFree);
+		CASE_CAT_NAME(EPagingPageOutFree);
+		CASE_CAT_NAME(EPagingRejuvenate);
+		CASE_CAT_NAME(EPagingPageNop);
+		CASE_CAT_NAME(EPagingPageLock);
+		CASE_CAT_NAME(EPagingPageUnlock);
+		CASE_CAT_NAME(EPagingPageOutCache);
+		CASE_CAT_NAME(EPagingPageInCode);
+		CASE_CAT_NAME(EPagingPageOutCode);
+		CASE_CAT_NAME(EPagingMapCode);
+		CASE_CAT_NAME(EPagingAged);
+		CASE_CAT_NAME(EPagingDecompressStart);
+		CASE_CAT_NAME(EPagingDecompressEnd);
+		CASE_CAT_NAME(EPagingMemoryModel);
+		CASE_CAT_NAME(EPagingChunkDonatePage);
+		CASE_CAT_NAME(EPagingChunkReclaimPage);
+		CASE_CAT_NAME(EPagingPageIn);
+		CASE_CAT_NAME(EPagingPageOut);
+		CASE_CAT_NAME(EPagingMapPage);
+		CASE_CAT_NAME(EPagingDonatePage);
+		CASE_CAT_NAME(EPagingReclaimPage);
+		CASE_CAT_NAME(EPagingAgedClean);
+		CASE_CAT_NAME(EPagingAgedDirty);
+		CASE_CAT_NAME(EPagingPageTableAlloc);
+			}
+		break;
+
+	case BTrace::EKernelMemory:
+		switch((BTrace::TKernelMemory)aSubCategory)
+			{
+		CASE_CAT_NAME(EKernelMemoryInitialFree);
+		CASE_CAT_NAME(EKernelMemoryCurrentFree);
+		CASE_CAT_NAME(EKernelMemoryMiscAlloc);
+		CASE_CAT_NAME(EKernelMemoryMiscFree);
+		CASE_CAT_NAME(EKernelMemoryDemandPagingCache);
+		CASE_CAT_NAME(EKernelMemoryDrvPhysAlloc);
+		CASE_CAT_NAME(EKernelMemoryDrvPhysFree);
+			}
+		break;
+
+	case BTrace::EMetaTrace:
+		{
+		if(aSubCategory==KJunkTraceSubcategory)
+			return "*JUNK*";
+		else
+			{
+			switch((BTrace::TMetaTrace)aSubCategory)
+				{
+			CASE_CAT_NAME(EMetaTraceTimestampsInfo);
+			CASE_CAT_NAME(EMetaTraceMeasurementStart);
+			CASE_CAT_NAME(EMetaTraceMeasurementEnd);
+			CASE_CAT_NAME(EMetaTraceFilterChange);
+				}
+			}
+		}
+		break;
+
+	case BTrace::EFastMutex:
+		switch((BTrace::TFastMutex)aSubCategory)
+			{
+		CASE_CAT_NAME(EFastMutexWait);
+		CASE_CAT_NAME(EFastMutexSignal);
+		CASE_CAT_NAME(EFastMutexFlash);
+		CASE_CAT_NAME(EFastMutexName);
+		CASE_CAT_NAME(EFastMutexBlock);
+			}
+		break;
+
+	case BTrace::EProfiling:
+		switch((BTrace::TProfiling)aSubCategory)
+			{
+		CASE_CAT_NAME(ECpuFullSample);
+		CASE_CAT_NAME(ECpuOptimisedSample);
+		CASE_CAT_NAME(ECpuIdfcSample);
+		CASE_CAT_NAME(ECpuNonSymbianThreadSample);
+			}
+		break;
+
+	case BTrace::ESymbianKernelSync:
+		switch((BTrace::TSymbianKernelSync)aSubCategory)
+			{
+		CASE_CAT_NAME(ESemaphoreCreate);
+		CASE_CAT_NAME(ESemaphoreDestroy);
+		CASE_CAT_NAME(ESemaphoreAcquire);
+		CASE_CAT_NAME(ESemaphoreRelease);
+		CASE_CAT_NAME(ESemaphoreBlock);
+		CASE_CAT_NAME(EMutexCreate);
+		CASE_CAT_NAME(EMutexDestroy);
+		CASE_CAT_NAME(EMutexAcquire);
+		CASE_CAT_NAME(EMutexRelease);
+		CASE_CAT_NAME(EMutexBlock);
+		CASE_CAT_NAME(ECondVarCreate);
+		CASE_CAT_NAME(ECondVarDestroy);
+		CASE_CAT_NAME(ECondVarBlock);
+		CASE_CAT_NAME(ECondVarWakeUp);
+		CASE_CAT_NAME(ECondVarSignal);
+		CASE_CAT_NAME(ECondVarBroadcast);
+			}
+		break;
+
+	case BTrace::EFlexibleMemModel:
+		switch((BTrace::TFlexibleMemModel)aSubCategory)
+			{
+		CASE_CAT_NAME(EMemoryObjectCreate);
+		CASE_CAT_NAME(EMemoryObjectDestroy);
+		CASE_CAT_NAME(EMemoryMappingCreate);
+		CASE_CAT_NAME(EMemoryMappingDestroy);
+		CASE_CAT_NAME(EMemoryObjectIsChunk);
+		CASE_CAT_NAME(EMemoryObjectIsCodeSeg);
+		CASE_CAT_NAME(EMemoryObjectIsProcessStaticData);
+		CASE_CAT_NAME(EMemoryObjectIsDllStaticData);
+		CASE_CAT_NAME(EMemoryObjectIsSupervisorStack);
+		CASE_CAT_NAME(EMemoryObjectIsUserStack);
+		CASE_CAT_NAME(EAddressSpaceId);
+			}
+		break;
+
+		}
+	return UnknownNames[aSubCategory];
+	}
+
+
+
+//
+// Data structures
+//
+
+enum TDataType
+	{
+	EDataTypeNone = 0,
+	EDataTypeText,
+	EDataTypeObject,
+	};
+
+class Thread;
+class Cpu;
+
+struct TraceHeader
+	{
+	TUint8	iCpuNum;
+	TUint8	iCategory;
+	TUint8	iSubCategory;
+	TUint8	iFlags;
+	TUint32	iHeader2;
+	TUint64	iTimestamp;
+	TUint32	iTimestamp2;
+	Thread*	iContextID;
+	TUint32	iPC;
+	TUint32	iExtra;
+	TUint8	iDataTypes[4];
+	TUint32	iCalculatedData[2];
+	TUint	iDataSize;
+	Cpu*	iCpu;
+	TUint32	iError;
+	};
+
+
+struct TraceRecord : public TraceHeader
+	{
+	TUint32 iData[(8+KMaxBTraceDataArray)/4];
+	};
+
+
+
+//
+// ECpuUsage traces
+//
+
+enum TContext
+	{
+	EContextThread,
+	EContextFiq,
+	EContextIrq,
+	EContextIDFC,
+	EContextUnknown
+	};
+
+class Cpu
+	{
+public:
+	Cpu();
+	void ChangeContext(TContext aType, Thread* aThread=0);
+	void Reset();
+
+	TContext	iCurrentContext;
+	TInt		iFiqNest;
+	TInt		iIrqNest;
+	TInt		iIDFCNest;
+	TUint64		iFiqTime;
+	TUint64		iIrqTime;
+	TUint64		iIDFCTime;
+	Thread*		iCurrentThread;
+	TUint64		iBaseTimestamp;
+	};
+
+//
+// Objects
+//
+
+const TUint KMaxTraceNameLength = 10;
+
+class Object
+	{
+public:
+	Object(TUint32 aTraceId, const char* aTraceNamePrefix)
+		: iTraceId(aTraceId), iIndex(~0u), iOwner(0), iOwnerTraceId(0), iAlive(1), iNameSet(false), iNameLength(0),
+		  iTraceNamePrefix(aTraceNamePrefix)
+		{
+		iName[0] = 0;
+		}
+
+	void Destroy()
+		{
+		if(iAlive)
+			--iAlive;
+		if(iOwnerTraceId && !iOwner)
+			--UnknownOwners;
+		}
+
+	virtual ~Object()
+		{}
+
+	void SetName(void* aName, TUint aLength)
+		{
+		ASSERT(aLength<sizeof(iName));
+		iNameLength = (TUint8)aLength;
+		memcpy(iName,aName,aLength);
+		iName[aLength] = 0;
+		iNameSet = true;
+		}
+
+	void SetName(TraceRecord& aTrace, TUint aIndex)
+		{
+		SetName(aTrace.iData+aIndex,aTrace.iDataSize-aIndex*4);
+		aTrace.iDataTypes[aIndex] = EDataTypeText;
+		}
+
+	TBool IsName(void* aName, TUint aLength)
+		{
+		if(aLength!=iNameLength)
+			return false;
+		while(aLength--)
+			if(iName[aLength]!=((TUint8*)aName)[aLength])
+				return false;
+		return true;
+		}
+
+	typedef TUint8 TraceNameBuf[KMaxTraceNameLength+1];
+	typedef TUint8 FullNameBuf[KMaxBTraceDataArray+2+KMaxBTraceDataArray+2+KMaxBTraceDataArray+1]; // space for name1::name2::name3[tracename]
+	typedef TUint8 FullTraceNameBuf[KMaxBTraceDataArray+1+KMaxBTraceDataArray+2+KMaxBTraceDataArray+KMaxTraceNameLength+1+1]; // space for [tracename]'name1::name2::name3'
+
+	TUint FullName(FullNameBuf& aName)
+		{
+		TUint length = 0;
+		if(iOwner)
+			{
+			if(iOwner->iOwner)
+				{
+				memcpy(aName+length,iOwner->iOwner->iName,iOwner->iOwner->iNameLength);
+				length += iOwner->iOwner->iNameLength;
+				aName[length++] = ':';
+				aName[length++] = ':';
+				}
+			memcpy(aName+length,iOwner->iName,iOwner->iNameLength);
+			length += iOwner->iNameLength;
+			aName[length++] = ':';
+			aName[length++] = ':';
+			}
+		memcpy(aName+length,iName,iNameLength);
+		length += iNameLength;
+		aName[length] = 0;
+		return length;
+		}
+
+	TUint TraceName(TraceNameBuf& aName)
+		{
+		TInt i = 0;
+		const TUint KNumDigits = KMaxTraceNameLength-4;
+		aName[i++] = '<';
+		const char* prefix = iTraceNamePrefix;
+		if(prefix[0])
+			aName[i++] = *prefix++;
+		if(prefix[0])
+			aName[i++] = *prefix++;
+		TUint n = iIndex;
+		for(TUint d=KNumDigits; d>0; --d)
+			{
+			aName[i+d-1] = TUint8('0'+(n%10));
+			n /= 10;
+			}
+		i += KNumDigits;
+		aName[i++] = '>';
+		aName[i] = 0;
+		return i;
+		}
+
+	TUint FullTraceName(FullTraceNameBuf& aName)
+		{
+		TUint l1 = TraceName(*(TraceNameBuf*)aName);
+		aName[l1++] = '\'';
+		TUint l2 = FullName(*(FullNameBuf*)(aName+l1));
+		aName[l1+l2++] = '\'';
+		aName[l1+l2] = 0;
+		return l1+l2;
+		}
+
+public:
+	TUint32 iTraceId;					///< ID for object as found in raw trace data.
+	TUint iIndex;						///< Index into container for this object.
+	Object* iOwner;						///< Object which 'owns' this one, e.g. process which owns a thread.
+	TUint32 iOwnerTraceId;				///< Trace ID for owner if owner object as yet unknown
+	TUint8 iAlive;						///< True if object destroyed trace not yet parsed.
+	TUint8 iNameSet;					///< True if name has been set.
+	TUint8 iNameLength;				
+	TUint8 iName[KMaxBTraceDataArray+1];	
+	const char* iTraceNamePrefix;
+public:
+	static TUint32 UnknownOwners;
+	};
+TUint32 Object::UnknownOwners = 0;
+
+
+class ObjectContainer
+	{
+public:
+	ObjectContainer()
+		: iNumEntries(0), iEntriesLength(0) , iEntries(0)
+		{
+		iLink = iAllContainers;
+		iAllContainers = this;
+		}
+
+	static void Reset()
+		{
+		ObjectContainer* container = iAllContainers;
+		while(container)
+			{
+			TUint i = container->iNumEntries;
+			while(i--)
+				free(container->iEntries[i].iItem);
+			free(container->iEntries);
+			container->iEntries = 0;
+			container->iNumEntries = 0;
+			container->iEntriesLength = 0;
+			container = container->iLink;
+			}
+		}
+
+	void Add(Object* aObject)
+		{
+		if(iNumEntries>=iEntriesLength)
+			{
+			iEntriesLength += 128;
+			iEntries = (Entry*)realloc(iEntries,iEntriesLength*sizeof(Entry));
+			ASSERT(iEntries);
+			}
+		aObject->iIndex = iNumEntries;
+		Entry& entry = iEntries[iNumEntries++];
+		entry.iTraceId = aObject->iTraceId;
+		entry.iItem = aObject;
+		}
+
+	TUint Count()
+		{
+		return iNumEntries;
+		}
+
+	Object* operator[](TInt aIndex)
+		{
+		if(TUint(aIndex)<iNumEntries)
+			return iEntries[aIndex].iItem;
+		ASSERT(0);
+		return 0;
+		}
+
+	Object* Find(TUint32 aTraceId)
+		{
+		Entry* ptr = iEntries+iNumEntries;
+		Entry* end = iEntries;
+		while(ptr>end)
+			{
+			--ptr;
+			if(ptr->iTraceId==aTraceId)
+				{
+				if(ptr->iItem->iAlive)
+					return ptr->iItem;
+				else
+					break;
+				}
+			}
+		return 0;
+		}
+private:
+	struct Entry
+		{
+		TUint32 iTraceId;
+		Object* iItem;
+		};
+	TUint iNumEntries;
+	TUint iEntriesLength;
+	Entry* iEntries;
+	ObjectContainer* iLink;
+
+	static ObjectContainer* iAllContainers;
+	};
+ObjectContainer* ObjectContainer::iAllContainers = 0;
+
+
+#define GENERIC_OBJECT_DEFINITIONS(C)							\
+																\
+	static C* Find(TUint32 aTraceId)							\
+		{														\
+		return (C*)iContainer.Find(aTraceId);					\
+		}														\
+																\
+	static C* Create(TraceRecord& aTrace, TUint aIndex)			\
+		{														\
+		TUint32 traceId = aTrace.iData[aIndex];					\
+		C* object = new C(traceId);								\
+		aTrace.iDataTypes[aIndex] = EDataTypeObject;			\
+		aTrace.iData[aIndex] = (uintptr_t)object;				\
+		return object;											\
+		}														\
+																\
+	static C* Find(TraceRecord& aTrace, TUint aIndex)			\
+		{														\
+		TUint32 traceId = aTrace.iData[aIndex];					\
+		C* object = Find(traceId);								\
+		if(!object)												\
+			return 0;											\
+		aTrace.iDataTypes[aIndex] = EDataTypeObject;			\
+		aTrace.iData[aIndex] = (uintptr_t)object;				\
+		return object;											\
+		}														\
+																\
+	static C* FindOrCreate(TraceRecord& aTrace, TUint aIndex)	\
+		{														\
+		C* object = Find(aTrace,aIndex);						\
+		if(!object)												\
+			object = Create(aTrace,aIndex);						\
+		return object;											\
+		}
+
+
+
+//
+// Process
+//
+
+class Process : public Object
+	{
+public:
+	Process(TUint32 aTraceId)
+		: Object(aTraceId,"P"), iThreadCount(0), iMaxThreadCount(0)
+		{
+		iContainer.Add(this);
+		}
+
+	GENERIC_OBJECT_DEFINITIONS(Process);
+
+public:
+	TUint iThreadCount;
+	TUint iMaxThreadCount;
+
+	static ObjectContainer iContainer;
+	};
+ObjectContainer Process::iContainer;
+
+
+
+//
+// Thread
+//
+class FastMutex;
+class Thread : public Object
+	{
+public:
+	Thread(TUint32 aTraceId)
+		: Object(aTraceId,"T"), iId(~0u), iCpuTime(0), iSamples(0)
+		{
+		iContainer.Add(this);
+		iNameLength = (TUint8)MakeName(iName,"NThread-",aTraceId);
+		}
+
+	TUint64 CpuTime()
+		{
+		if(!iLastCpu || !iLastCpu->iBaseTimestamp)
+			return 0;
+		if(iLastCpu->iCurrentThread==this)
+			return iCpuTime + Timestamp - iLastCpu->iBaseTimestamp;
+		return iCpuTime;
+		}
+
+	void Sampled()
+		{
+		if( iSamples+1 != 0xFFFFFFFF)
+			{
+			iSamples++;
+			}
+		}
+
+	GENERIC_OBJECT_DEFINITIONS(Thread);
+
+	static Object* FindThreadOrProcess(TraceRecord& aTrace, TUint aIndex)
+		{
+		if (!aTrace.iData[aIndex])
+			return 0;
+		Object* p = Find(aTrace, aIndex);
+		if (!p)
+			p = Process::Find(aTrace, aIndex);
+		return p;
+		}
+public:
+	TUint32 iId;
+	Cpu* iLastCpu;
+	TUint64 iCpuTime;
+	TUint64 iFMBlockStartTime;
+	FastMutex* iWaitFastMutex;
+
+	// Number of profiling samples
+	TUint32 iSamples;
+
+	static ObjectContainer iContainer;
+	};
+
+ObjectContainer Thread::iContainer;
+
+
+
+//
+// Chunk
+//
+
+TUint ChunkErrors = 0;
+
+const TUint KPageShift = 12; // chunk memory is allocated in 4kB pages
+
+class Chunk : public Object
+	{
+public:
+	Chunk(TUint32 aTraceId)
+		: Object(aTraceId,"C"), iCurrentSize(0), iPeakSize(0), iMaxSize(0), iPageMap(0), iTraceErrors(0)
+		{
+		iContainer.Add(this);
+		}
+
+	~Chunk()
+		{
+		free(iPageMap);
+		}
+
+	void SetMaxSize(TUint32 aMaxSize)
+		{
+		ASSERT(!iMaxSize);
+		iMaxSize = aMaxSize;
+		TUint mapSize = ((aMaxSize>>KPageShift)+7)>>3;
+		iPageMap = (TUint8*)malloc(mapSize);
+		ASSERT(iPageMap);
+		memset(iPageMap,0,mapSize);
+		iCurrentSize = 0;
+		}
+
+	void Commit(TUint32 aStart,TUint32 aSize)
+		{
+		if(!iPageMap) // we havent been intialised yet
+			return;
+
+		if(aStart+aSize<aStart || aStart+aSize>iMaxSize)
+			{
+			++iTraceErrors;
+			++ChunkErrors;
+			return;
+			}
+		if(SetBits(iPageMap,aStart>>KPageShift,aSize>>KPageShift))
+			{
+			++iTraceErrors;
+			++ChunkErrors;
+			ErrorOnThisTrace = true;
+			}
+		}
+
+	void Decommit(TUint32 aStart,TUint32 aSize)
+		{
+		if(!iPageMap) // we havent been intialised yet
+			return;
+
+		// Decommit is complicated by the fact that aSize is the number of pages
+		// actually decommited, which may be less than the region of the original
+		// chunk decommit operation. E.g. if pages 1 and 3 were originally decommited
+		// and the decommit operation was for pages 0-3, then the trace has a size of
+		// 2 pages, even though the operation was on 4 pages.
+		// We handle this, repeatedly decommiting from our iPageMap, until we have
+		// freed aSize bytes worth of pages...
+		while(aSize)
+			{
+			if(aStart+aSize<aStart || aStart+aSize>iMaxSize)
+				{
+				// we haven't found enough memory to decommit
+				++iTraceErrors;
+				++ChunkErrors;
+				ErrorOnThisTrace = true;
+				return;
+				}
+			TUint notDecommitted = ClearBits(iPageMap,aStart>>KPageShift,aSize>>KPageShift);
+			aStart += aSize;
+			aSize = notDecommitted<<KPageShift;
+			}
+		}
+
+	void ResetMemory()
+		{
+		}
+
+	GENERIC_OBJECT_DEFINITIONS(Chunk);
+
+public:
+	TUint32 iCurrentSize;
+	TUint32 iPeakSize;
+	TUint32 iMaxSize;
+	TUint8* iPageMap;
+	TUint iTraceErrors;
+
+	static ObjectContainer iContainer;
+	};
+ObjectContainer Chunk::iContainer;
+
+
+
+//
+// Semaphore, Mutex, CondVar
+//
+class Semaphore : public Object
+	{
+public:
+	Semaphore(TUint32 aTraceId)
+		: Object(aTraceId,"S")
+		{
+		iContainer.Add(this);
+		}
+
+	~Semaphore()
+		{
+		}
+
+	GENERIC_OBJECT_DEFINITIONS(Semaphore);
+public:
+
+	static ObjectContainer iContainer;
+	};
+ObjectContainer Semaphore::iContainer;
+
+
+class Mutex : public Object
+	{
+public:
+	Mutex(TUint32 aTraceId)
+		: Object(aTraceId,"M")
+		{
+		iContainer.Add(this);
+		}
+
+	~Mutex()
+		{
+		}
+
+	GENERIC_OBJECT_DEFINITIONS(Mutex);
+public:
+
+	static ObjectContainer iContainer;
+	};
+ObjectContainer Mutex::iContainer;
+
+
+class CondVar : public Object
+	{
+public:
+	CondVar(TUint32 aTraceId)
+		: Object(aTraceId,"V")
+		{
+		iContainer.Add(this);
+		}
+
+	~CondVar()
+		{
+		}
+
+	GENERIC_OBJECT_DEFINITIONS(CondVar);
+public:
+
+	static ObjectContainer iContainer;
+	};
+ObjectContainer CondVar::iContainer;
+
+
+
+
+//
+// CodeSeg
+//
+
+TUint CodeSegErrors = 0;
+
+class CodeSeg : public Object
+	{
+public:
+	CodeSeg(TUint32 aTraceId)
+		: Object(aTraceId,"CS"), iAllocatedMemory(0)
+		{
+		iContainer.Add(this);
+		}
+
+	GENERIC_OBJECT_DEFINITIONS(CodeSeg);
+
+	TUint iAllocatedMemory;
+public:
+	static ObjectContainer iContainer;
+	};
+ObjectContainer CodeSeg::iContainer;
+
+
+
+//
+// FastMutex
+//
+
+TUint FastMutexNestErrors = 0;
+
+class FastMutex : public Object
+	{
+public:
+	FastMutex(TUint32 aTraceId)
+		: Object(aTraceId,"FM"), iHoldingThread(0), iHeldCount(0), iTotalHeldTime(0),
+		  iMaxHeldTime(0), iMaxHeldTimestamp(0), iMaxHeldPc(0), iBlockCount(0)
+		{
+		iContainer.Add(this);
+		iNameLength = (TUint8)MakeName(iName,"NFastMutex-",aTraceId);
+		iNameLength = 19;
+		}
+
+	TUint32 Wait(Thread* aThread)
+		{
+		TUint32 time = 0;
+		if(iHoldingThread)
+			{
+			++FastMutexNestErrors;
+			ErrorOnThisTrace = true;
+			}
+		iHoldingThread = aThread;
+		iWaitCpuTimeBase = aThread->CpuTime();
+		if (aThread->iWaitFastMutex == this)
+			{
+			time = (TUint32)(Timestamp - aThread->iFMBlockStartTime);
+			}
+		aThread->iWaitFastMutex = 0;
+		return time;
+		}
+
+	void Block(Thread* aThread)
+		{
+		if (aThread->iWaitFastMutex != this)
+			{
+			aThread->iFMBlockStartTime = Timestamp;
+			aThread->iWaitFastMutex = this;
+			++iBlockCount;
+			}
+		};
+
+	TUint32 Signal(Thread* aThread, TUint32 aPc)
+		{
+		if (!iHoldingThread)	// First record for this mutex so ignore it as don't
+			return 0;			// have waiting thread details
+		
+		if(iHoldingThread!=aThread)
+			{
+			++FastMutexNestErrors;
+			ErrorOnThisTrace = true;
+			iHoldingThread = 0;
+			return 0;
+			}
+		iHoldingThread = 0;
+		TUint64 time = aThread->CpuTime()-iWaitCpuTimeBase;
+		++iHeldCount;
+		iTotalHeldTime += time;
+		if(time>iMaxHeldTime)
+			{
+			iMaxHeldTime = time;
+			iMaxHeldPc = aPc;
+			iMaxHeldTimestamp = Timestamp;
+			}
+		return (TUint32)time;
+		}
+
+	GENERIC_OBJECT_DEFINITIONS(FastMutex);
+
+public:
+	Thread* iHoldingThread;
+	TUint32 iHeldCount;		// number of times mutex acquired
+	TUint64 iTotalHeldTime;
+	TUint64 iWaitCpuTimeBase;
+	TUint64 iMaxHeldTime;
+	TUint64 iMaxHeldTimestamp;
+	TUint32 iMaxHeldPc;
+	TUint32 iBlockCount;	// number of times mutex caused a thread to wait
+
+	static ObjectContainer iContainer;
+	};
+ObjectContainer FastMutex::iContainer;
+
+
+
+
+//
+// ProfilingSample
+//
+
+TUint ProfilingSampleErrors = 0;
+
+class ProfilingSample : public Object
+	{
+public:
+	ProfilingSample(TUint32 aTraceId)
+		: Object(aTraceId,"PS")
+		{
+		iContainer.Add(this);
+		iNameLength = (TUint8)MakeName(iName,"ProfilingSample-",aTraceId);
+		}
+
+	void SetPC( TUint32 aPC )
+		{
+		iPC = aPC;
+		}
+
+	void SetThread( TUint32 aThread )
+		{
+		if( 0 != aThread )
+			iThread = aThread;
+		}
+
+	void SetType(const BTrace::TProfiling aType)
+		{
+		iType = aType;
+		}
+
+	GENERIC_OBJECT_DEFINITIONS(ProfilingSample);
+
+
+public:
+
+	static ObjectContainer iContainer;
+	static Thread* iTopTen[10];
+	static TUint32 iSamples;
+	static TUint32 iLastThread;
+
+	TUint32 iPC;
+	TUint32 iThread;
+	BTrace::TProfiling iType;
+
+	};
+
+TUint32 ProfilingSample::iSamples = 0;
+TUint32 ProfilingSample::iLastThread = 0;
+
+ObjectContainer ProfilingSample::iContainer;
+
+
+//
+// EThreadIdentification traces
+//
+
+void PreProcessThreadIdentification(TraceRecord& aTrace)
+	{
+	Thread* thread;
+	Process* process;
+
+	switch((BTrace::TThreadIdentification)aTrace.iSubCategory)
+		{
+	case BTrace::ENanoThreadCreate:
+		CHECK_TRACE_DATA_WORDS(1);
+		thread = Thread::FindOrCreate(aTrace,0);
+		break;
+
+	case BTrace::ENanoThreadDestroy:
+		CHECK_TRACE_DATA_WORDS(1);
+		thread = Thread::Find(aTrace,0);
+		if(thread)
+			thread->Destroy();
+		break;
+
+	case BTrace::EThreadCreate:
+	case BTrace::EThreadName:
+		CHECK_TRACE_DATA_WORDS(2);
+		thread = Thread::FindOrCreate(aTrace,0);
+		if(aTrace.iSubCategory==BTrace::EThreadCreate)
+			++thread->iAlive; // thread needs destroying twice (ENanoThreadDestroy+EThreadDestroy)
+		process = Process::FindOrCreate(aTrace,1);
+		thread->iOwner = process;
+		++process->iThreadCount;
+		if(process->iThreadCount>process->iMaxThreadCount)
+			process->iMaxThreadCount = process->iThreadCount;
+		thread->SetName(aTrace,2);
+		break;
+
+	case BTrace::EThreadDestroy:
+		CHECK_TRACE_DATA_WORDS(1);
+		thread = Thread::Find(aTrace,0);
+		if(thread)
+			{
+			thread->Destroy();
+			process = (Process*)thread->iOwner;
+			if(process && process->iThreadCount)
+				--process->iThreadCount;
+			}
+		break;
+
+	case BTrace::EProcessName:
+		CHECK_TRACE_DATA_WORDS(2);
+		if(aTrace.iData[0])
+			{
+			thread = Thread::FindOrCreate(aTrace,0);
+			process = Process::Find(aTrace.iData[1]);
+			if(!process || (process->iNameLength && !process->IsName(aTrace.iData+2,aTrace.iDataSize-2*4)))
+				{
+				if(process)
+					process->Destroy();
+				process = Process::Create(aTrace,1); // no existing process, or name different
+				}
+			else
+				process = Process::Find(aTrace,1); // find again (this will update trace data[1])
+			}
+		else
+			process = Process::Find(aTrace,1);
+		if(process)
+			process->SetName(aTrace,2);
+		break;
+
+	case BTrace::EThreadId:
+		CHECK_TRACE_DATA_WORDS(2);
+		thread = Thread::FindOrCreate(aTrace,0);
+		process = Process::FindOrCreate(aTrace,1);
+		thread->iId = aTrace.iData[2];
+		break;
+
+	case BTrace::EProcessCreate:
+		CHECK_TRACE_DATA_WORDS(1);
+		process = Process::FindOrCreate(aTrace,0);
+		break;
+
+	case BTrace::EProcessDestroy:
+		CHECK_TRACE_DATA_WORDS(1);
+		process = Process::FindOrCreate(aTrace,0);
+		if(process)
+			process->Destroy();
+		break;
+
+	default:
+		break;
+		}
+	}
+
+
+//
+// ECpuUsage traces
+//
+
+Cpu TheCpus[KMaxCpus];
+TUint InterruptNestErrors = 0;
+TUint CpuUsagePresent = 0;
+
+Cpu::Cpu()
+	:	iCurrentContext(EContextUnknown),
+		iFiqNest(0),
+		iIrqNest(0),
+		iIDFCNest(0),
+		iFiqTime(0),
+		iIrqTime(0),
+		iIDFCTime(0),
+		iCurrentThread(0),
+		iBaseTimestamp(0)
+	{
+	}
+
+void Cpu::Reset()
+	{
+	iCurrentContext = EContextUnknown;
+	iFiqNest = 0;
+	iIrqNest = 0;
+	iIDFCNest = 0;
+	iCurrentThread = 0;
+	iBaseTimestamp = 0;
+	}
+
+void ResetCpuUsage()
+	{
+	TInt i;
+	for (i=0; i<KMaxCpus; ++i)
+		TheCpus[i].Reset();
+	}
+
+
+void StartCpuUsage()
+	{
+	TInt i;
+	for (i=0; i<KMaxCpus; ++i)
+		new (&TheCpus[i]) Cpu;
+	InterruptNestErrors = 0;
+	}
+
+
+void Cpu::ChangeContext(TContext aType, Thread* aThread)
+	{
+	TUint64 delta = Timestamp-iBaseTimestamp;
+	switch(iCurrentContext)
+		{
+	case EContextThread:
+		iCurrentThread->iCpuTime += delta;
+		break;
+	case EContextFiq:
+		iFiqTime += delta;
+		break;
+	case EContextIrq:
+		iIrqTime += delta;
+		break;
+	case EContextIDFC:
+		iIDFCTime += delta;
+		break;
+	default:
+		break;
+		}
+
+	if(aType==EContextUnknown)
+		{
+		if(iFiqNest)
+			aType = EContextFiq;
+		else if(iIrqNest)
+			aType = EContextIrq;
+		else if(iIDFCNest)
+			aType = EContextIDFC;
+		else
+			{
+			aType = EContextThread;
+			aThread = iCurrentThread;
+			}
+		}
+
+	if(aType==EContextThread && !aThread)
+		{
+		iCurrentContext = EContextUnknown;
+		iBaseTimestamp = 0;
+		return;
+		}
+
+	iCurrentContext = aType;
+	if(aType==EContextThread)
+		{
+		iCurrentThread = aThread;
+		aThread->iLastCpu = this;
+		}
+
+	iBaseTimestamp = Timestamp;
+	}
+
+
+void PreProcessCpuUsage(TraceRecord& aTrace)
+	{
+	CpuUsagePresent = true;
+	Cpu& cpu = *aTrace.iCpu;
+
+	switch((BTrace::TCpuUsage)aTrace.iSubCategory)
+		{
+	case BTrace::EIrqStart:
+		++cpu.iIrqNest;
+		cpu.ChangeContext(EContextIrq);
+		break;
+
+	case BTrace::EFiqStart:
+		++cpu.iFiqNest;
+		cpu.ChangeContext(EContextFiq);
+		break;
+
+	case BTrace::EIDFCStart:
+		if(cpu.iIrqNest+cpu.iFiqNest>1 || cpu.iIDFCNest!=0)
+			{
+			++InterruptNestErrors;
+			ErrorOnThisTrace = true;
+			}
+		cpu.iIrqNest = 0;
+		cpu.iFiqNest = 0;
+		cpu.iIDFCNest = 1;
+		cpu.ChangeContext(EContextIDFC);
+		break;
+
+	case BTrace::EIrqEnd:
+		if(cpu.iIrqNest)
+			--cpu.iIrqNest;
+		else
+			{
+			++InterruptNestErrors;
+			ErrorOnThisTrace = true;
+			}
+		cpu.ChangeContext(EContextUnknown);
+		break;
+
+	case BTrace::EFiqEnd:
+		if(cpu.iFiqNest)
+			--cpu.iFiqNest;
+		else
+			{
+			++InterruptNestErrors;
+			ErrorOnThisTrace = true;
+			}
+		cpu.ChangeContext(EContextUnknown);
+		break;
+
+	case BTrace::EIDFCEnd:
+		if(cpu.iIDFCNest!=1)
+			{
+			++InterruptNestErrors;
+			ErrorOnThisTrace = true;
+			}
+		cpu.iIDFCNest = 0;
+		cpu.ChangeContext(EContextUnknown);
+		break;
+
+	case BTrace::ENewThreadContext:
+		if(cpu.iIrqNest+cpu.iFiqNest>1 || cpu.iIDFCNest!=0)
+			{
+			++InterruptNestErrors;
+			ErrorOnThisTrace = true;
+			}
+		cpu.iIrqNest = 0;
+		cpu.iFiqNest = 0;
+		cpu.iIDFCNest = 0;
+		cpu.ChangeContext(EContextThread,aTrace.iContextID);
+		break;
+		}
+	}
+
+
+void ReportThreads()
+	{
+	TUint numThreads = Thread::iContainer.Count();
+	if(!numThreads)
+		return;
+
+	TUint64 totalTime = 0;
+	printf("\nREPORT: Threads\n\n");
+	WarnIfError(0);
+	printf("%-10s %5s %10s %8s %s\n","","State","CPUTime","TraceId","Name");
+	TUint i;
+	for(i=0; i<numThreads; ++i)
+		{
+		Thread* thread = (Thread*)Thread::iContainer[i];
+		Object::FullNameBuf fullName;
+		thread->FullName(fullName);
+		Object::TraceNameBuf name;
+		thread->TraceName(name);
+		printf("%-10s %5s %10d %08x '%s'\n",name,
+			thread->iAlive?(const char*)"Alive":(const char*)"Dead",
+			(int)Time(thread->iCpuTime),(int)thread->iTraceId,fullName);
+		totalTime += thread->iCpuTime;
+		}
+	for (i=0; i<(TUint)KMaxCpus; ++i)
+		{
+		printf("CPU %1d\n", i);
+		Cpu& cpu = TheCpus[i];
+		printf("%-10s %5s %10d %s\n","FIQ","",(int)Time(cpu.iFiqTime),"");
+		printf("%-10s %5s %10d %s\n","IRQ","",(int)Time(cpu.iIrqTime),"");
+		printf("%-10s %5s %10d %s\n","IDFC","",(int)Time(cpu.iIDFCTime),"");
+		totalTime += cpu.iFiqTime + cpu.iIrqTime + cpu.iIDFCTime;
+		}
+	printf("%-10s %5s ----------\n","","");
+	printf("%-10s %5s %10d\n","","",(int)Time(totalTime));
+	printf("\n");
+	}
+
+
+void ReportProcesses()
+	{
+	TUint numProcesses = Process::iContainer.Count();
+	if(!numProcesses)
+		return;
+
+	printf("\nREPORT: Processes\n\n");
+	WarnIfError(0);
+	printf("%-10s %5s %7s %8s %s\n","","State","Threads","TraceId","Name");
+	TUint i;
+	for(i=0; i<numProcesses; ++i)
+		{
+		Process* process = (Process*)Process::iContainer[i];
+		Object::FullNameBuf fullName;
+		process->FullName(fullName);
+		Object::TraceNameBuf name;
+		process->TraceName(name);
+		printf("%-10s %5s %3u/%-3u %08x '%s'\n",name,
+			process->iAlive?(const char*)"Alive":(const char*)"Dead",
+			(unsigned int)process->iThreadCount,(unsigned int)process->iMaxThreadCount,
+			(unsigned int)process->iTraceId,fullName);
+		}
+	printf("\n");
+	}
+
+
+void EndCpuUsage()
+	{
+	TInt i;
+	for (i=0; i<KMaxCpus; ++i)
+		TheCpus[i].ChangeContext(EContextUnknown);
+	}
+
+
+//
+// EChunks traces
+//
+
+void StartChunks()
+	{
+	ChunkErrors = 0;
+	}
+
+
+void PreProcessChunks(TraceRecord& aTrace)
+	{
+	CHECK_TRACE_DATA_WORDS(1);
+	Chunk* chunk = Chunk::FindOrCreate(aTrace,0);
+
+	switch((BTrace::TChunks)aTrace.iSubCategory)
+		{
+	case BTrace::EChunkCreated:
+		CHECK_TRACE_DATA_WORDS(2);
+		chunk->SetName(aTrace,2);
+		chunk->SetMaxSize(aTrace.iData[1]);
+		// start by assuming thread is 'owned' by the thread which created it...
+		chunk->iOwner = aTrace.iContextID;
+		break;
+
+	case BTrace::EChunkInfo:
+		CHECK_TRACE_DATA_WORDS(3);
+		break; // ignore
+
+	case BTrace::EChunkDestroyed:
+		chunk->Destroy();
+		break;
+
+	case BTrace::EChunkMemoryAllocated:
+		{
+		CHECK_TRACE_DATA_WORDS(3);
+		chunk->Commit(aTrace.iData[1],aTrace.iData[2]);
+		TUint32 size = chunk->iCurrentSize+aTrace.iData[2];
+		if(size<chunk->iCurrentSize || size>chunk->iMaxSize)
+			size = chunk->iMaxSize;
+		chunk->iCurrentSize = size;
+		if(size>chunk->iPeakSize)
+			chunk->iPeakSize = size;
+		aTrace.iCalculatedData[0] = size/1024;
+		}
+		break;
+
+	case BTrace::EChunkMemoryDeallocated:
+		{
+		CHECK_TRACE_DATA_WORDS(3);
+		chunk->Decommit(aTrace.iData[1],aTrace.iData[2]);
+		TUint32 size = chunk->iCurrentSize-aTrace.iData[2];
+		if(size>chunk->iCurrentSize)
+			size = 0;
+		chunk->iCurrentSize = size;
+		aTrace.iCalculatedData[0] = size/1024;
+		}
+		break;
+
+	case BTrace::EChunkMemoryAdded:
+		CHECK_TRACE_DATA_WORDS(3);
+		chunk->Commit(aTrace.iData[1],aTrace.iData[2]);
+		break;
+
+	case BTrace::EChunkMemoryRemoved:
+		CHECK_TRACE_DATA_WORDS(3);
+		chunk->Decommit(aTrace.iData[1],aTrace.iData[2]);
+		break;
+
+	case BTrace::EChunkOwner:
+		{
+		CHECK_TRACE_DATA_WORDS(2);
+		Process* process = Process::FindOrCreate(aTrace,1);
+		// set owner, unless current owner is owned by the same process
+		// (this preserves creating thread names in ownership list which is more useful)
+		if(!chunk->iOwner || chunk->iOwner->iOwner!=process)
+			chunk->iOwner = process;
+		}
+		break;
+
+		}
+	}
+
+
+void ReportChunks()
+	{
+	TUint numChunks = Chunk::iContainer.Count();
+	if(!numChunks)
+		return;
+
+	if(!ReportLevel)
+		printf("\nREPORT: Chunks (Named objects only)\n\n");
+	else
+		printf("\nREPORT: Chunks\n\n");
+	WarnIfError(ChunkErrors);
+	printf("%-10s %5s %8s %8s %8s %8s %s\n",
+		"","State","Size","Peak","Max","TraceId","Name");
+	TUint totalSize = 0;
+	TUint i;
+	for(i=0; i<numChunks; ++i)
+		{
+		Chunk* chunk = (Chunk*)Chunk::iContainer[i];
+		if(ReportLevel==0 && !chunk->iNameSet)
+			continue; // only report explicitly named mutexes at report level 0
+		Object::FullNameBuf fullName;
+		chunk->FullName(fullName);
+		Object::TraceNameBuf name;
+		chunk->TraceName(name);
+		printf("%-10s %5s %7uk %7uk %7uk %08x '%s'\n",
+			name,chunk->iAlive?(const char*)"Alive":(const char*)"Dead",
+			(unsigned int)chunk->iCurrentSize/1024,(unsigned int)chunk->iPeakSize/1024,(unsigned int)chunk->iMaxSize/1024,
+			(unsigned int)chunk->iTraceId,fullName);
+		totalSize += chunk->iCurrentSize/1024;
+		}
+	printf("%-10s %5s --------\n","","");
+	printf("%-10s %5s %7uk\n","","",totalSize);
+	printf("\n");
+	}
+
+
+
+//
+// CodeSeg
+//
+
+void StartCodeSegs()
+	{
+	CodeSegErrors = 0;
+	}
+
+
+void PreProcessCodeSegs(TraceRecord& aTrace)
+	{
+	CHECK_TRACE_DATA_WORDS(1);
+	CodeSeg* codeseg;
+
+	switch((BTrace::TCodeSegs)aTrace.iSubCategory)
+		{
+	case BTrace::ECodeSegCreated:
+		codeseg = CodeSeg::FindOrCreate(aTrace,0);
+		codeseg->SetName(aTrace,1);
+		break;
+
+	case BTrace::ECodeSegInfo:
+		CHECK_TRACE_DATA_WORDS(10);
+		CodeSeg::FindOrCreate(aTrace,0);
+/*		- 4 bytes containing the attributes.
+		- 4 bytes containing the code base address (.text).
+		- 4 bytes containing the size of the code section (.text).
+		- 4 bytes containing the base address of the constant data section (.radata).
+		- 4 bytes containing the size of the constant data section (.radata).
+		- 4 bytes containing the base address of the initialised data section (.data).
+		- 4 bytes containing the size of the initialised data section (.data).
+		- 4 bytes containing the base address of the uninitialised data section (.bss).
+		- 4 bytes containing the size of the uninitialised data section (.bss).
+*/		break;
+
+	case BTrace::ECodeSegDestroyed:
+		codeseg = CodeSeg::FindOrCreate(aTrace,0);
+		codeseg->Destroy();
+		codeseg->iAllocatedMemory = 0; // clear this now because ECodeSegMemoryDeallocated comes after codeseg destroy
+		break;
+
+	case BTrace::ECodeSegMapped:
+		CHECK_TRACE_DATA_WORDS(2);
+		codeseg = CodeSeg::FindOrCreate(aTrace,0);
+		Process::FindOrCreate(aTrace,1);
+		break;
+
+	case BTrace::ECodeSegUnmapped:
+		CHECK_TRACE_DATA_WORDS(2);
+		CodeSeg::FindOrCreate(aTrace,0);
+		Process::FindOrCreate(aTrace,1);
+		break;
+
+	case BTrace::ECodeSegMemoryAllocated:
+		CHECK_TRACE_DATA_WORDS(2);
+		codeseg = CodeSeg::FindOrCreate(aTrace,0);
+		codeseg->iAllocatedMemory += aTrace.iData[1];
+		if(codeseg->iAllocatedMemory<aTrace.iData[1])
+			{
+			codeseg->iAllocatedMemory = ~0u; // overflowed!
+			++CodeSegErrors;
+			ErrorOnThisTrace = true;
+			}
+		break;
+
+	case BTrace::ECodeSegMemoryDeallocated:
+		{
+		CHECK_TRACE_DATA_WORDS(2);
+		codeseg = CodeSeg::Find(aTrace,0);
+		if(codeseg)
+			{
+			TUint32 memory = codeseg->iAllocatedMemory-aTrace.iData[1];
+			if(memory>codeseg->iAllocatedMemory)
+				{
+				memory = 0; // underflowed
+				++CodeSegErrors;
+				ErrorOnThisTrace = true;
+				}
+			codeseg->iAllocatedMemory = memory;
+			}
+		}
+		break;
+
+		}
+	}
+
+
+void ReportCodeSegs()
+	{
+	TUint numCodeSegs = CodeSeg::iContainer.Count();
+	if(!numCodeSegs)
+		return;
+
+	if(!ReportLevel)
+		printf("\nREPORT: CodeSegs (Named objects only)\n\n");
+	else
+		printf("\nREPORT: CodeSegs\n\n");
+	WarnIfError(CodeSegErrors);
+	printf("%-10s %5s %8s %8s %s\n",
+		"","State","Memory","TraceId","Name");
+	TUint totalSize = 0;
+	TUint i;
+	for(i=0; i<numCodeSegs; ++i)
+		{
+		CodeSeg* codeseg = (CodeSeg*)CodeSeg::iContainer[i];
+		if(ReportLevel==0 && !codeseg->iNameSet)
+			continue; // only report explicitly named mutexes at report level 0
+		Object::FullNameBuf fullName;
+		codeseg->FullName(fullName);
+		Object::TraceNameBuf name;
+		codeseg->TraceName(name);
+		printf("%-10s %5s %7uk %08x '%s'\n",
+			name,codeseg->iAlive?(const char*)"Alive":(const char*)"Dead",
+			(unsigned int)codeseg->iAllocatedMemory/1024,(unsigned int)codeseg->iTraceId,fullName);
+		totalSize += codeseg->iAllocatedMemory/1024;
+		}
+	printf("%-10s %5s --------\n","","");
+	printf("%-10s %5s %7uk\n","","",totalSize);
+	printf("\n");
+	}
+
+
+
+//
+// MetaTrace
+//
+
+TUint KernelMemoryInitialFree = 0;
+TUint KernelMemoryCurrentFree = 0;
+TUint KernelMemoryMisc = 0;
+TUint KernelMemoryDrvPhys = 0;
+TUint KernelMemoryDemandPagingCache = 0;
+TUint KernelMemoryErrors = 0;
+TBool KernelMemoryTracesPresent = false;
+
+void StartKernelMemory()
+	{
+	KernelMemoryInitialFree = 0;
+	KernelMemoryCurrentFree = 0;
+	KernelMemoryMisc = 0;
+	KernelMemoryDrvPhys = 0;
+	KernelMemoryErrors = 0;
+	KernelMemoryTracesPresent = false;
+	}
+
+
+void PreProcessKernelMemory(TraceRecord& aTrace)
+	{
+	CHECK_TRACE_DATA_WORDS(1);
+	KernelMemoryTracesPresent = true;
+	switch((BTrace::TKernelMemory)aTrace.iSubCategory)
+		{
+	case BTrace::EKernelMemoryInitialFree:
+		KernelMemoryInitialFree = aTrace.iData[0];
+		aTrace.iCalculatedData[0] = KernelMemoryInitialFree/1024;
+		break;
+
+	case BTrace::EKernelMemoryCurrentFree:
+		KernelMemoryCurrentFree = aTrace.iData[0];
+		aTrace.iCalculatedData[0] = KernelMemoryCurrentFree/1024;
+		break;
+
+	case BTrace::EKernelMemoryMiscAlloc:
+		KernelMemoryMisc += aTrace.iData[0];
+		if(KernelMemoryMisc < aTrace.iData[0])
+			{
+			KernelMemoryMisc = 0xffffffffu;
+			++KernelMemoryErrors;
+			ErrorOnThisTrace = true;
+			}
+		aTrace.iCalculatedData[0] = KernelMemoryMisc/1024;
+		break;
+
+	case BTrace::EKernelMemoryMiscFree:
+		if(KernelMemoryMisc >= aTrace.iData[0])
+			KernelMemoryMisc -= aTrace.iData[0];
+		else
+			{
+			KernelMemoryMisc = 0;
+			++KernelMemoryErrors;
+			ErrorOnThisTrace = true;
+			}
+		aTrace.iCalculatedData[0] = KernelMemoryMisc/1024;
+		break;
+
+	case BTrace::EKernelMemoryDemandPagingCache:
+		KernelMemoryDemandPagingCache = aTrace.iData[0];
+		break;
+
+	case BTrace::EKernelMemoryDrvPhysAlloc:
+		KernelMemoryDrvPhys += aTrace.iData[0];
+		if(KernelMemoryDrvPhys < aTrace.iData[0])
+			{
+			KernelMemoryDrvPhys = 0xffffffffu;
+			++KernelMemoryErrors;
+			ErrorOnThisTrace = true;
+			}
+		aTrace.iCalculatedData[0] = KernelMemoryDrvPhys/1024;
+		break;
+
+	case BTrace::EKernelMemoryDrvPhysFree:
+		if(KernelMemoryDrvPhys >= aTrace.iData[0])
+			KernelMemoryDrvPhys -= aTrace.iData[0];
+		else
+			{
+			KernelMemoryDrvPhys = 0;
+			++KernelMemoryErrors;
+			ErrorOnThisTrace = true;
+			}
+		aTrace.iCalculatedData[0] = KernelMemoryDrvPhys/1024;
+		break;
+		}
+	}
+
+void ReportKernelMemory()
+	{
+	if(!KernelMemoryTracesPresent)
+		return;
+
+	printf("\nREPORT: Kernel Memory\n\n");
+	WarnIfError(KernelMemoryErrors);
+	printf("Total RAM size............................. %dk\n",KernelMemoryInitialFree/1024);
+	printf("Miscelaneous RAM used by kernel............ %dk\n",KernelMemoryMisc/1024);
+	printf("Physical RAM allocated by device drivers... %dk\n",KernelMemoryDrvPhys/1024);
+	printf("Demand paging cache reserve................ %dk\n",KernelMemoryDemandPagingCache/1024);
+	if(ReportLevel>1)
+		printf("Last 'current free RAM' value seen......... %dk\n",KernelMemoryCurrentFree/1024);
+
+	printf("\n");
+	}
+
+//
+// MetaTrace
+//
+
+void StartMetaTrace()
+	{
+	TimestampPeriod = 0;
+	Timestamp2Period = 0;
+	}
+
+
+void PreProcessMetaTrace(TraceRecord& aTrace)
+	{
+	switch((BTrace::TMetaTrace)aTrace.iSubCategory)
+		{
+	case BTrace::EMetaTraceTimestampsInfo:
+		CHECK_TRACE_DATA_WORDS(3);
+		TimestampPeriod = aTrace.iData[0];
+		Timestamp2Period = aTrace.iData[1];
+		Timestamp64Bit = aTrace.iData[2]&1;
+		break;
+
+	case BTrace::EMetaTraceMeasurementStart:
+	case BTrace::EMetaTraceMeasurementEnd:
+		CHECK_TRACE_DATA_WORDS(2);
+		aTrace.iDataTypes[2] = EDataTypeText;
+		break;
+
+	case BTrace::EMetaTraceFilterChange:
+		CHECK_TRACE_DATA_WORDS(1);
+		break;
+		}
+	}
+
+//
+// EFastMutex traces
+//
+
+void StartFastMutex()
+	{
+	FastMutexNestErrors = 0;
+	}
+
+
+void PreProcessFastMutex(TraceRecord& aTrace)
+	{
+	CHECK_TRACE_DATA_WORDS(1);
+	FastMutex* mutex = FastMutex::FindOrCreate(aTrace,0);
+	Thread* thread = aTrace.iContextID;
+
+	switch((BTrace::TFastMutex)aTrace.iSubCategory)
+		{
+	case BTrace::EFastMutexWait:
+		aTrace.iCalculatedData[0] = Time(mutex->Wait(thread));
+		break;
+
+	case BTrace::EFastMutexSignal:
+		aTrace.iCalculatedData[0] = Time(mutex->Signal(thread,aTrace.iPC));
+		break;
+
+	case BTrace::EFastMutexFlash:
+		aTrace.iCalculatedData[0] = Time(mutex->Signal(thread,aTrace.iPC));
+		mutex->Wait(thread);
+		break;
+
+	case BTrace::EFastMutexName:
+		CHECK_TRACE_DATA_WORDS(2);
+		mutex->SetName(aTrace,2);
+		break;
+
+	case BTrace::EFastMutexBlock:
+		mutex->Block(thread);
+		break;
+
+		}
+	}
+
+
+void PreProcessSymbianKernelSync(TraceRecord& aTrace)
+	{
+	switch((BTrace::TSymbianKernelSync)aTrace.iSubCategory)
+		{
+	case BTrace::ESemaphoreCreate:
+		{
+		CHECK_TRACE_DATA_WORDS(2);
+		TUint32 ownerid = aTrace.iData[1];
+		Semaphore* sem = Semaphore::FindOrCreate(aTrace,0);
+		Object* owner = Thread::FindThreadOrProcess(aTrace,1);
+		sem->iOwner = owner;
+		if (!owner && ownerid)
+			sem->iOwnerTraceId = ownerid, ++Object::UnknownOwners;
+		sem->SetName(aTrace,2);
+		break;
+		}
+
+	case BTrace::ESemaphoreDestroy:
+		{
+		CHECK_TRACE_DATA_WORDS(1);
+		Semaphore* sem = Semaphore::Find(aTrace,0);
+		if (sem)
+			sem->Destroy();
+		break;
+		}
+
+	case BTrace::ESemaphoreAcquire:
+	case BTrace::ESemaphoreRelease:
+	case BTrace::ESemaphoreBlock:
+		{
+		CHECK_TRACE_DATA_WORDS(1);
+		Semaphore::FindOrCreate(aTrace,0);
+		break;
+		}
+
+
+	case BTrace::EMutexCreate:
+		{
+		CHECK_TRACE_DATA_WORDS(2);
+		TUint32 ownerid = aTrace.iData[1];
+		Mutex* m = Mutex::FindOrCreate(aTrace,0);
+		Object* owner = Thread::FindThreadOrProcess(aTrace,1);
+		m->iOwner = owner;
+		if (!owner && ownerid)
+			m->iOwnerTraceId = ownerid, ++Object::UnknownOwners;
+		m->SetName(aTrace,2);
+		break;
+		}
+
+	case BTrace::EMutexDestroy:
+		{
+		CHECK_TRACE_DATA_WORDS(1);
+		Mutex* m = Mutex::Find(aTrace,0);
+		if (m)
+			m->Destroy();
+		break;
+		}
+
+	case BTrace::EMutexAcquire:
+	case BTrace::EMutexRelease:
+	case BTrace::EMutexBlock:
+		{
+		CHECK_TRACE_DATA_WORDS(1);
+		Mutex::FindOrCreate(aTrace,0);
+		break;
+		}
+
+
+	case BTrace::ECondVarCreate:
+		{
+		CHECK_TRACE_DATA_WORDS(2);
+		TUint32 ownerid = aTrace.iData[1];
+		CondVar* cv = CondVar::FindOrCreate(aTrace,0);
+		Object* owner = Thread::FindThreadOrProcess(aTrace,1);
+		cv->iOwner = owner;
+		if (!owner && ownerid)
+			cv->iOwnerTraceId = ownerid, ++Object::UnknownOwners;
+		cv->SetName(aTrace,2);
+		break;
+		}
+
+	case BTrace::ECondVarDestroy:
+		{
+		CHECK_TRACE_DATA_WORDS(1);
+		CondVar* cv = CondVar::Find(aTrace,0);
+		if (cv)
+			cv->Destroy();
+		break;
+		}
+
+	case BTrace::ECondVarBlock:
+	case BTrace::ECondVarWakeUp:
+	case BTrace::ECondVarSignal:
+	case BTrace::ECondVarBroadcast:
+		{
+		CHECK_TRACE_DATA_WORDS(1);
+		CondVar::FindOrCreate(aTrace,0);
+		break;
+		}
+
+
+	default:
+		break;
+		}
+	}
+
+
+void ReportFastMutex()
+	{
+	TUint numMutexes = FastMutex::iContainer.Count();
+	if(!numMutexes)
+		return;
+
+	if(!ReportLevel)
+		printf("\nREPORT: FastMutexes (Named objects only)\n\n");
+	else
+		printf("\nREPORT: FastMutexes\n\n");
+	WarnIfError(0);
+	printf("%-10s %8s %8s %10s %10s %-8s %12s %8s %s\n",
+		"","MaxTime","AveTime","HeldCount","BlockCount","MaxPC","MaxTimestamp","TraceId","Name");
+	TUint i;
+	for(i=0; i<numMutexes; ++i)
+		{
+		FastMutex* mutex = (FastMutex*)FastMutex::iContainer[i];
+		if(ReportLevel==0 && !mutex->iNameSet)
+			continue; // only report explicitly named mutexes at report level 0
+		Object::FullNameBuf fullName;
+		mutex->FullName(fullName);
+		Object::TraceNameBuf name;
+		mutex->TraceName(name);
+		TUint32 averageHeldTime = mutex->iHeldCount ? Time(mutex->iTotalHeldTime/mutex->iHeldCount) : 0;
+		printf("%-10s %8u %8u %10u %10u %08x %12u %08x '%s'\n",
+			name,(unsigned int)Time(mutex->iMaxHeldTime),(unsigned int)averageHeldTime,(unsigned int)mutex->iHeldCount,(unsigned int)mutex->iBlockCount,
+			(unsigned int)mutex->iMaxHeldPc,(unsigned int)Time(mutex->iMaxHeldTimestamp-TimestampBase),(unsigned int)mutex->iTraceId,fullName);
+		}
+	printf("\n");
+	}
+
+
+//
+// ProfilingSample
+//
+
+void StartProfilingSample()
+	{
+	ProfilingSampleErrors = 0;
+	}
+
+
+/** 
+	Index 0 of TraceRecord is the program counter
+	The only one not having it are ECpuNonSymbianThreadSample samples.
+	Index 1 of TraceRecord is the NThread pointer.
+	The only one that has it is ECpuFullSample.
+	The samples are identified by their index, which is maintained by 
+	ProfilingSample::iSamples. Thus to create a ProfilingSample object we
+	need to put this value in the data at index 0 after copying the PC 
+	and Thread id (if present).
+	The reasoning is that all samples need to be represented and thus we 
+	need to create a ProfilingSample object, even when they are on the same 
+	PC and or thread. Each sample important and should not be discarded.
+*/
+void PreProcessProfiling(TraceRecord& aTrace)
+	{
+
+	ProfilingSample* sample;
+	Thread* thread;
+
+	switch((BTrace::TProfiling)aTrace.iSubCategory)
+		{
+
+	case BTrace::ECpuFullSample:
+		{
+		CHECK_TRACE_DATA_WORDS(2);
+
+		TUint32 aThread = aTrace.iData[1];
+		// The thread id is aTrace.iData[1], so find or create it
+		// This action can modify aTrace.iData[1], that is why we took a copy above
+		thread = Thread::FindOrCreate(aTrace,1);
+		if( thread )
+			thread->Sampled();
+
+		TUint32 aPC = aTrace.iData[0];
+
+		// Always create a sample identified by the running counter ProfilingSample::iSamples
+		aTrace.iData[0] = ProfilingSample::iSamples;
+		sample = ProfilingSample::Create(aTrace,0);
+		if( sample )
+			{
+			sample->SetPC( aPC );
+			sample->SetThread( aThread );
+			sample->SetType(BTrace::ECpuFullSample);
+			}
+
+		ProfilingSample::iLastThread = aThread;
+		}
+		break;
+
+	case BTrace::ECpuOptimisedSample:
+		{
+		CHECK_TRACE_DATA_WORDS(1);
+		TUint32 aPC = aTrace.iData[0];
+
+		aTrace.iData[0] = ProfilingSample::iSamples;
+		sample = ProfilingSample::Create(aTrace,0);
+		if( sample )
+			{
+			sample->SetPC( aPC );
+			sample->SetType( BTrace::ECpuOptimisedSample );
+			sample->SetThread(ProfilingSample::iLastThread);
+			}
+
+		if( 0 != ProfilingSample::iLastThread )
+			{
+			thread = Thread::Find(ProfilingSample::iLastThread);
+			if( thread )
+				{
+				thread->Sampled();
+				}
+			}
+
+		}
+		break;
+
+	case BTrace::ECpuIdfcSample:
+		{
+		CHECK_TRACE_DATA_WORDS(1);
+		TUint32 aPC = aTrace.iData[0];
+
+		aTrace.iData[0] = ProfilingSample::iSamples;
+		sample = ProfilingSample::Create(aTrace,0);
+
+		sample->SetPC( aPC );
+		sample->SetType(BTrace::ECpuIdfcSample);
+
+		}
+		break;
+
+	case BTrace::ECpuNonSymbianThreadSample:
+		{
+		// No data
+		aTrace.iData[0] = ProfilingSample::iSamples;
+		sample = ProfilingSample::Create(aTrace,0);
+		sample->SetType(BTrace::ECpuNonSymbianThreadSample);
+
+		}
+		break;
+
+	default:
+		ProfilingSampleErrors++;
+		ErrorOnThisTrace = true;
+		}
+
+	ProfilingSample::iSamples++;
+
+	}
+
+
+void ReportSampleProfiling()
+	{
+	printf("\nREPORT: Profiling\n\n");
+
+	TUint numSamples = ProfilingSample::iContainer.Count();
+	if(!numSamples)
+		{
+		printf("\n        No Samples\n\n");
+		return;
+		}
+
+	WarnIfError(0);
+
+
+	// Print thread samples
+	TUint numThreads = Thread::iContainer.Count();
+	if(numThreads)
+		{
+		printf(" Samples by Thread\n\n");
+		printf("%-11s %-8s %-8s\t%-12s\t%s\n\n", "", "TraceId", "Samples", "%", "Name");
+		TUint i;
+		TReal threadPercentage;
+		for(i=0; i<numThreads; ++i)
+			{
+			Thread* thread = (Thread*)Thread::iContainer[i];
+
+			if( thread && thread->iSamples )
+				{
+				Object::FullNameBuf fullName;
+				thread->FullName(fullName);
+				Object::TraceNameBuf name;
+				thread->TraceName(name);
+
+				threadPercentage = thread->iSamples*100.0/numSamples;
+
+				printf("%-10s %08x %8d\t%02.2f\t'%s'\n",
+							name,
+							(unsigned int)thread->iTraceId,
+							(unsigned int)(thread->iSamples),
+							threadPercentage,
+							fullName );
+
+				}//if samples
+			}//for numThreads
+		}//if threads
+
+
+	if(ReportLevel>0)
+		{
+
+		printf("\nAll samples\n\n%-21s %-8s %-8s\n\n", "Type", "ThreadId", "PC");
+
+		TUint i;
+		TUint fullSamples = 0;
+		TUint optSamples = 0;
+		TUint dfcSamples = 0;
+		TUint nonSymbSamples = 0;
+
+		for(i=0; i<numSamples; ++i)
+			{
+			ProfilingSample* sample = (ProfilingSample*)ProfilingSample::iContainer[i];
+			switch((BTrace::TProfiling)sample->iType)
+				{
+				case BTrace::ECpuFullSample:
+					{
+					if( ReportLevel>1)
+						printf("ECpuFull              %08x %08x\n",
+							 (unsigned int)(sample->iThread), (unsigned int)(sample->iPC) );
+
+					fullSamples++;
+					}
+					break;
+				case BTrace::ECpuOptimisedSample:
+					{
+					if( ReportLevel>1)
+						printf("ECpuOptimised         %08x %08x\n",
+							 (unsigned int)(sample->iThread), (unsigned int)(sample->iPC) );
+
+					optSamples++;
+					}
+					break;
+				case BTrace::ECpuIdfcSample:
+					{
+					if( ReportLevel>1)
+						printf("ECpuIdfc              %08x\n", (unsigned int)(sample->iPC) );
+
+					dfcSamples++;
+					}
+					break;
+				case BTrace::ECpuNonSymbianThreadSample:
+					{
+					if( ReportLevel>1)
+						printf("ECpuNonSymbianThread\n");
+
+					nonSymbSamples++;
+					}
+					break;
+				}//switch
+			}//for
+
+
+		TReal typePercentage;
+
+		printf("\nSamples by type\n");
+
+		typePercentage = fullSamples * 100.0 / numSamples;
+		printf(" Samples of type ECpuFullSample :\t\t%-10d\t%02.2f %%\n", fullSamples, typePercentage  );
+
+		typePercentage = optSamples * 100.0 / numSamples;
+		printf(" Samples of type ECpuOptimisedSample :\t\t%-10d\t%02.2f %%\n", optSamples, typePercentage  );
+
+		typePercentage = dfcSamples * 100.0 / numSamples;
+		printf(" Samples of type ECpuIdfcSample :\t\t%-10d\t%02.2f %%\n", dfcSamples, typePercentage  );
+
+		typePercentage = nonSymbSamples * 100.0 / numSamples;
+		printf(" Samples of type ECpuNonSymbianThreadSample :\t%-10d\t%02.2f %%\n", nonSymbSamples, typePercentage  );
+
+		printf(" Total Samples : \t\t\t\t%d\n", numSamples );
+
+		}//report level
+
+	printf("\n");
+	}
+
+//
+// Trace processing
+//
+
+TraceRecord** TraceIndex = 0;
+TUint TraceIndexSize = 0;
+TUint32 NextTraceId = 0;
+TraceRecord* LastTrace = 0;
+TBool Timestamp2Present = 0;
+TBool TraceDumpStarted = false;
+
+
+void StartTrace()
+	{
+	TraceDumpStarted = false;
+	TraceFormatErrors = 0;
+	TraceBufferFilled = false;
+	Timestamp2Present = false;
+	}
+
+
+TUint32 ReadTraceWord(const TUint8*& header)
+	{
+	TUint32 word;
+	memcpy(&word, header, sizeof(TUint32));
+	header += sizeof(TUint32);
+	return word;
+	}
+
+
+TBool PreProcessTrace(TraceRecord& aTrace, const TUint8* aData)
+	{
+	ErrorOnThisTrace = false;
+	aTrace.iError = 0;
+
+	aTrace.iDataSize = 0; // initialise to safe value
+
+	// process aTrace header...
+	TUint traceSize = aData[BTrace::ESizeIndex];
+	if(traceSize<4u || traceSize>(TUint)KMaxBTraceRecordSize)
+		{
+		aTrace.iError = 1;
+		return false; // bad size
+		}
+	aTrace.iCpuNum = 0;
+
+	TUint8 flags = aData[BTrace::EFlagsIndex];
+	if(!TraceRecordId)						// first trace record...?
+		flags &= ~BTrace::EMissingRecord;	// ignore missing traces before log start
+	aTrace.iFlags = flags;
+
+	TUint8 category = aData[BTrace::ECategoryIndex];
+	aTrace.iCategory = category;
+
+	TUint8 subCategory = aData[BTrace::ESubCategoryIndex];
+	aTrace.iSubCategory = subCategory;
+
+	const TUint8* header = aData+4;
+
+	TUint32 header2 = 0;
+	if(flags&BTrace::EHeader2Present)
+		{
+		header2 = ReadTraceWord(header);
+		aTrace.iCpuNum = (TUint8)(header2>>20);
+		}
+	aTrace.iHeader2 = header2;
+	aTrace.iCpu = TheCpus + aTrace.iCpuNum;
+
+	// process timestamp and timestamp2...
+	TUint32 ts1 = 0;
+	TUint32 ts2 = 0;
+	TUint64 timestamp = 0;
+	if(flags&BTrace::ETimestampPresent)
+		ts1 = ReadTraceWord(header);
+	if(flags&BTrace::ETimestamp2Present)
+		{
+		Timestamp2Present = true;
+		ts2 = ReadTraceWord(header);
+		}
+	aTrace.iTimestamp2 = ts2;
+	if(flags&BTrace::ETimestampPresent)
+		{
+		if (Timestamp64Bit)
+			{
+			timestamp = ts2;
+			timestamp <<= 32;
+			timestamp |= ts1;
+			Timestamp = timestamp;
+			}
+		else
+			{
+			timestamp = ts1;
+			if(timestamp<(Timestamp&0xffffffffu))
+				Timestamp += TUint64(1)<<32;
+			Timestamp &= TUint64(0xffffffff)<<32;
+			Timestamp |= timestamp; 
+			timestamp = Timestamp;
+			}
+		if(!TraceRecordId)
+			TimestampBase = timestamp; // record timestamp of first trace
+		}
+	aTrace.iTimestamp = timestamp;
+
+	// process context...
+	// coverity[assign_zero]
+	aTrace.iContextID = 0;
+	if(flags&BTrace::EContextIdPresent)
+		{
+		TUint32 contextId = ReadTraceWord(header);
+		Thread* thread = Thread::Find(contextId);
+		if(!thread)
+			thread = new Thread(contextId);
+		aTrace.iContextID = thread;
+		}
+
+	// process pc...
+	TUint32 pc = 0;
+	if(flags&BTrace::EPcPresent)
+		pc = ReadTraceWord(header);
+	aTrace.iPC = pc;
+
+	// process extra...
+	TUint32 extra = 0;
+	if(flags&BTrace::EExtraPresent)
+		extra = ReadTraceWord(header);
+	aTrace.iExtra = extra;
+
+	// process payload data...
+	TUint headerSize = header-aData;
+	aData = (TUint8*)header;
+	if(headerSize>traceSize)
+		{
+		aTrace.iError = 1;
+		return false; // bad trace record
+		}
+	TUint dataSize = traceSize-headerSize;
+	if(dataSize>sizeof(aTrace.iData))
+		{
+		aTrace.iError = 1;
+		return false; // bad trace record
+		}
+	aTrace.iDataSize = dataSize;
+	memcpy(&aTrace.iData,aData,dataSize);
+
+	// clear pre-processor specific data...
+	aTrace.iDataTypes[0] = 0;
+	aTrace.iDataTypes[1] = 0;
+	aTrace.iDataTypes[2] = 0;
+	aTrace.iDataTypes[3] = 0;
+	aTrace.iCalculatedData[0] = 0;
+	aTrace.iCalculatedData[1] = 0;
+
+	// check for missing.
+	if(flags & BTrace::EMissingRecord)
+		{// Some trace was missing as the btrace buffer must have been filled.
+		TraceBufferFilled = true;
+		aTrace.iError = 1;
+		return false;
+		}
+
+	// category specific processing...
+	switch(aTrace.iCategory)
+		{
+	case BTrace::ERDebugPrintf:
+	case BTrace::EKernPrintf:
+	case BTrace::EPlatsecPrintf:
+		if((flags&BTrace::EHeader2Present) && (header2&BTrace::EMultipartFlagMask))
+			aTrace.iDataTypes[2] = EDataTypeText;
+		else
+			aTrace.iDataTypes[1] = EDataTypeText;
+		break;
+	case BTrace::EThreadIdentification:
+		PreProcessThreadIdentification(aTrace); break;
+	case BTrace::ECpuUsage:
+		PreProcessCpuUsage(aTrace); break;
+	case BTrace::EChunks:
+		PreProcessChunks(aTrace); break;
+	case BTrace::ECodeSegs:
+		PreProcessCodeSegs(aTrace); break;
+	case BTrace::EKernelMemory:
+		PreProcessKernelMemory(aTrace); break;
+	case BTrace::EMetaTrace:
+		PreProcessMetaTrace(aTrace); break;
+	case BTrace::EFastMutex:
+		PreProcessFastMutex(aTrace); break;
+	case BTrace::EProfiling:
+		PreProcessProfiling(aTrace); break;
+	case BTrace::ESymbianKernelSync:
+		PreProcessSymbianKernelSync(aTrace); break;
+	default:
+		break;
+		}
+
+	// update trace ID...
+	++TraceRecordId;
+	if (ErrorOnThisTrace)
+		aTrace.iError = 1;
+	return true;
+	}
+
+
+void DumpTrace(TraceRecord& aTrace)
+	{
+	if(!TraceDumpStarted)
+		{
+		// print heading...
+		if(SMP)
+			printf("C ");
+		if(Timestamp2Present)
+			printf("%10s ","TimeStamp2");
+		printf("%10s ","Time");
+		printf("%-8s ","PC");
+		if(ReportLevel>2)
+			{
+			printf("%-60s ","Context");
+			printf("%18s ","Category");
+			printf("%24s ","SubCategory");
+			}
+		else
+			{
+			printf("%-10s ","Context");
+			printf("%24s ","SubCategory");
+			}
+		printf("Data...\n");
+		TraceDumpStarted = true;
+		}
+
+	if(aTrace.iFlags&BTrace::EMissingRecord)
+		printf("MISSING TRACE RECORD(S)\n");
+
+	// print CPU number
+	if (SMP)
+		{
+		printf("%1d ", aTrace.iCpuNum);
+		}
+
+	// print timestamp...
+	if(Timestamp2Present)
+		{
+		if(aTrace.iFlags&BTrace::ETimestamp2Present)
+			printf("%10u ",(unsigned int)aTrace.iTimestamp2);
+		else
+			printf("           ");
+		}
+
+	if(aTrace.iFlags&BTrace::ETimestampPresent)
+		printf("%10u ",(unsigned int)Time(aTrace.iTimestamp-TimestampBase));
+	else
+		printf("           ");
+
+	// print PC...
+	if(aTrace.iFlags&BTrace::EPcPresent)
+		printf("%08x ",(unsigned int)aTrace.iPC);
+	else
+		printf("         ");
+
+	// print context...
+	if(ReportLevel>2)
+		{
+		Object::FullTraceNameBuf fullName;
+		fullName[0] = 0;
+		if(aTrace.iFlags&BTrace::EContextIdPresent)
+			aTrace.iContextID->FullTraceName(fullName);
+		printf("%-60s ",fullName);
+		}
+	else
+		{
+		Object::TraceNameBuf traceName;
+		traceName[0] = 0;
+		if(aTrace.iFlags&BTrace::EContextIdPresent)
+			aTrace.iContextID->TraceName(traceName);
+		printf("%-10s ",traceName);
+		}
+	// print trace categories...
+	const char* catName = CategoryName(aTrace.iCategory);
+	const char* subCatName = SubCategoryName(aTrace.iCategory,aTrace.iSubCategory);
+	if(ReportLevel>2)
+		printf("%18s %-24s ",catName,subCatName);
+	else
+		{
+		if(subCatName[0])
+			printf("%24s ",subCatName);
+		else
+			printf("%24s ",catName);
+		};
+
+	// print trace data contents...
+	TUint i;
+	for(i=0; i<aTrace.iDataSize; i+=4)
+		{
+		TUint32 data = aTrace.iData[i/sizeof(TUint32)];
+		if(i<16)
+			{
+			// first 4 words of data may have 'type' info set during pre-processing...
+			switch(aTrace.iDataTypes[i/sizeof(TUint32)])
+				{
+			case EDataTypeObject:
+				{
+				// data is an object, print "full name"[traceID]...
+				Object* object = (Object*)data;
+				if(ReportLevel>2)
+					{
+					Object::FullTraceNameBuf name;
+					object->FullTraceName(name);
+					printf("%s ",name);
+					}
+				else
+					{
+					Object::TraceNameBuf name;
+					object->TraceName(name);
+					printf("%s ",name);
+					}
+				}
+				continue;
+
+			case EDataTypeText:
+				{
+				// rest of trace is text...
+				TUint8* text = (TUint8*)aTrace.iData+i;
+				TUint8* textEnd = text+(aTrace.iDataSize-i);
+				TUint8 buffer[256];
+				TUint x=0;
+				while(text<textEnd && x<sizeof(buffer)-2)
+					{
+					TUint8 c = *text++;
+					TUint8 escape = 0;
+					switch(c)
+						{
+					case 9: escape = 't'; break;
+					case 10: escape = 'n'; break;
+					case 13: escape = 'r'; break;
+					default:
+						if(c<' ') c = '?';
+						break;
+						}
+					if(!escape)
+						buffer[x++] = c;
+					else
+						{
+						buffer[x++] = '\\';
+						buffer[x++] = escape;
+						}
+					}
+				buffer[x] = 0;
+				printf("\"%s\" ",buffer);
+				i = aTrace.iDataSize; // skip to end of data
+				}
+				continue;
+
+			default:
+				break;
+				}
+			}
+		// default to print data as hex value...
+		printf("%08x ",(unsigned int)data);
+		}
+
+	// print any extra data added by pre-processing...
+	for(i=0; i<2; ++i)
+		{
+		if(aTrace.iCalculatedData[i])
+			printf("{%u} ",(unsigned int)aTrace.iCalculatedData[i]);
+		}
+
+	if (aTrace.iError)
+		printf(" ***ERROR***");
+
+	// end-of-line finally!
+	printf("\n");
+	}
+
+
+void DumpAllTraces()
+	{
+	printf("\nREPORT: Trace Dump\n\n");
+	for(TUint i=0; i<NextTraceId; i++)
+		DumpTrace(*TraceIndex[i]);
+	printf("\n");
+	}
+
+
+void ReportErrors()
+	{
+	TBool errors = 	TraceFormatErrors || InterruptNestErrors || TraceFormatErrors 
+		|| ChunkErrors || CodeSegErrors || FastMutexNestErrors || KernelMemoryErrors
+		|| ProfilingSampleErrors;
+
+	if(!errors)
+		return;
+		
+	printf("\nREPORT: Trace Analysis Errors\n\n");
+	if(TraceFormatErrors)
+		printf("\tTrace Format Errors = %d\n",TraceFormatErrors);
+	if(InterruptNestErrors)
+		printf("\tInterrupt Nest Errors = %d\n",InterruptNestErrors);
+	if(FastMutexNestErrors)
+		printf("\tFast Mutex Nest Errors = %d\n",FastMutexNestErrors);
+	if(KernelMemoryErrors)
+		printf("\tKernel Memory Errors = %d\n",KernelMemoryErrors);
+	if(ChunkErrors)
+		printf("\tChunk Errors = %d\n",ChunkErrors);
+	if(CodeSegErrors)
+		printf("\tCodeSeg Errors = %d\n",CodeSegErrors);
+	if(ProfilingSampleErrors)
+		printf("\tProfiling Errors = %d\n",ProfilingSampleErrors);
+	printf("\n");
+	}
+
+
+/**
+The last trace record created has been preporcessed.
+*/
+void DoneTrace()
+	{
+	if(LastTrace)
+		TraceIndex[NextTraceId++] = (TraceRecord*)realloc(LastTrace,sizeof(TraceHeader)+LastTrace->iDataSize);
+	LastTrace = 0;
+	}
+
+
+/**
+Create a new trace record.
+*/
+TraceRecord* NewTrace()
+	{
+	if(NextTraceId>=TraceIndexSize)
+		{
+		TraceIndexSize += 1024;
+		TraceIndex = (TraceRecord**)realloc(TraceIndex,TraceIndexSize*sizeof(TraceIndex[0]));
+		ASSERT(TraceIndex);
+		}
+	DoneTrace();
+	LastTrace = (TraceRecord*)malloc(sizeof(TraceRecord));
+	return LastTrace;
+	}
+
+
+/**
+Delete all processed traces records.
+*/
+void ResetTrace()
+	{
+	DoneTrace();
+	TUint i;
+	for(i=0; i<NextTraceId; ++i)
+		free(TraceIndex[i]);
+	free(TraceIndex);
+	TraceIndex = 0;
+	TraceIndexSize = 0;
+	LastTrace = 0;
+	NextTraceId = 0;
+	TraceRecordId = 0;
+	}
+
+
+void EndTrace()
+	{
+	DoneTrace();
+	}
+
+
+/**
+Process an entire BTrace log capture.
+
+@param aInput		Pointer to function which will supply raw trace data.
+					Function should place up to aMaxSize bytes of data at aBuffer and return
+					the number of bytes stored. Return zero to indicate end of data.
+@param aReportLevel	Level of detail required of trace alaysis.
+					0 = brief summary, 1 = full summary, 2 = condensed trace dump, 3 = full trace dump.
+*/
+void ProcessAllTrace(TUint (*aInput)(TAny* aBuffer, TUint aMaxSize),TInt aReportLevel)
+	{
+	ReportLevel = aReportLevel;
+//	__UHEAP_MARK;
+	printf("Btrace Analysis:\n");
+	printf("\nTHIS TOOL IS UNOFFICIAL, UNSUPPORTED AND SUBJECT TO CHANGE WITHOUT NOTICE!\n");
+
+	StartTrace();
+	StartCpuUsage();
+	StartChunks();
+	StartCodeSegs();
+	StartFastMutex();
+	StartKernelMemory();
+	StartMetaTrace();
+	StartProfilingSample();
+
+	for(; !TraceBufferFilled ;)
+		{
+		// read more data...
+		TUint size = (*aInput)(TraceBuffer+TraceBufferSize, sizeof(TraceBuffer)-TraceBufferSize);
+		if(!size)
+			break;
+		TraceBufferSize += size;
+
+		// process all the complete traces in buffer...
+		const TUint8* data = TraceBuffer;
+		TUint sizeRemaining = TraceBufferSize;
+		while(sizeRemaining>BTrace::ESizeIndex)
+			{
+			TUint traceSize = (data[BTrace::ESizeIndex]+3)&~3;
+			if(traceSize>sizeRemaining)
+				break;
+
+			TraceRecord* trace = NewTrace();
+			ASSERT(trace);
+			if(!PreProcessTrace(*trace,data))
+				{
+				if (!TraceBufferFilled)
+					{
+					// bad trace, create dummy 1 byte trace record...
+					memset(trace,0,sizeof(*trace));
+					trace->iCategory = BTrace::EMetaTrace;
+					trace->iSubCategory = KJunkTraceSubcategory;
+					trace->iDataSize = 4;
+					trace->iData[0] = *data;
+					++TraceFormatErrors;
+					ErrorOnThisTrace = true;
+					traceSize = 1;
+					}
+				else // The buffer was filled so ignore the rest of the data
+					break;
+				}
+
+			data += traceSize;
+			sizeRemaining -= traceSize;
+			}
+		
+		if (!TraceBufferFilled)
+			{
+			memcpy(TraceBuffer,data,sizeRemaining);
+			TraceBufferSize = sizeRemaining;
+			}
+		else
+			{
+			// The trace buffer was filled so ignore the rest of the data
+			// and just read whatever is left to flush it from the btrace buffer.
+			while ((*aInput)(TraceBuffer, sizeof(TraceBuffer))){};
+			TraceBufferSize = 0;	// reset here so a format error isn't reported
+			}
+
+		if(aReportLevel<2)
+			ResetTrace(); // free up memory as we go along
+		}
+	EndTrace();
+	EndCpuUsage();
+
+	if(TraceBufferSize)
+		{
+		++TraceFormatErrors;
+		ErrorOnThisTrace = true;
+		}
+
+	ReportTimeUnits();
+	ReportErrors();
+	if(aReportLevel>=2)
+		DumpAllTraces();
+	if(ReportLevel>=1 || CpuUsagePresent)
+		{
+		ReportProcesses();
+		ReportThreads();
+		}
+	ReportChunks();
+	ReportKernelMemory();
+	ReportCodeSegs();
+	ReportFastMutex();
+	ReportSampleProfiling();
+
+	ResetTrace();
+	ObjectContainer::Reset();
+//	__UHEAP_MARKEND;
+	}
+
+