os/graphics/windowing/windowserver/econs/D_EXC.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 1999-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include <e32std.h>
    17 #include <e32svr.h>
    18 #include <f32file.h>
    19 
    20 _LIT(KBlankLine, "\r");
    21 
    22 class TDll
    23 	{
    24 public:
    25 	TDll(const TDesC& aName, TUint8* aBase, TInt aSize)
    26 		: iName(aName), iBase((TUint)aBase), iSize((TUint)aSize)
    27 		{}
    28 
    29 	TFullName iName;
    30 	TUint iBase;
    31 	TUint iSize;
    32 	};
    33 
    34 class TDllList
    35 	{
    36 public:
    37 	static void FindDlls();
    38 	static void ResetAndDestroy();
    39 	static void ListDlls(TFileText& aTextFile);
    40 	static TInt Match(TUint anAddr, const TDll*& aDll);
    41 
    42 private:
    43 	static RPointerArray<TDll> iList;
    44 	};
    45 
    46 RPointerArray<TDll> TDllList::iList(10);
    47 
    48 void TDllList::ResetAndDestroy()
    49 	{
    50 	iList.ResetAndDestroy();
    51 	}
    52 
    53 void TDllList::ListDlls(TFileText& aTextFile)
    54 	{
    55 	TBuf<0x100> line;
    56 
    57 	_LIT(KDllFormat, "%08x-%08x %S\r");
    58 
    59 	aTextFile.Write(KBlankLine);
    60 	for (TInt i=0; i<iList.Count(); i++)
    61 		{
    62 		const TDll& dll=*(iList[i]);
    63 		line.Format(KDllFormat, dll.iBase, dll.iBase+dll.iSize-1, &dll.iName);
    64 		aTextFile.Write(line);
    65 		}
    66 	}
    67 
    68 TInt TDllList::Match(TUint anAddr, const TDll*& aDll)
    69 	{
    70 	for (TInt i=0; i<iList.Count(); i++)
    71 		{
    72 		aDll=iList[i];
    73 		if (aDll->iBase<=anAddr && aDll->iBase+aDll->iSize > anAddr)
    74 			return KErrNone;
    75 		}
    76 	return KErrNotFound;
    77 	}
    78 
    79 
    80 void TDllList::FindDlls()
    81 //
    82 // For each library known to the system, look for a chunk of the same name
    83 //
    84 	{
    85 	TFindLibrary findLib;
    86 	TFullName libName;
    87 
    88 	while (findLib.Next(libName)==KErrNone)
    89 		{
    90 		TFindChunk findChunk(libName);
    91 		TFullName chunkName;
    92 		if (findChunk.Next(chunkName)!=KErrNone)
    93 			continue;
    94 		RChunk chunk;
    95 		if (chunk.Open(findChunk)!=KErrNone)
    96 			continue;
    97 		TUint8* base=chunk.Base();
    98 		TInt size=chunk.Size();
    99 		chunk.Close();
   100 		TDll* dllptr=new TDll(libName,base,size);
   101 		if (dllptr!=0)
   102 			iList.Append(dllptr);
   103 		}
   104 	}
   105 
   106 _LIT(KFormatStackInfo,"Stack %08x-%08x (? %d?), sp=%08x\r");
   107 
   108 HBufC8* GrabStack(const TDesC& /*aLine1*/, TThreadId aId, TUint aSp, TInt& aStackBase, TInt& aStackSize)
   109 	{
   110 	TInt err=KErrNone;
   111 #if 0
   112 	// Defect in E32 162 which means that RThread::GetRamSizes will always reset the machine!
   113 	RThread thread;
   114 	if (thread.Open(aId)!=KErrNone)
   115 		return 0;
   116 
   117 	TInt heapsize=0;
   118 	err=thread.GetRamSizes(heapsize,aStackSize);
   119 	thread.Close();
   120 	
   121 	if (err!=KErrNone)
   122 		return 0;
   123 
   124 	// Sanity check
   125 	aStackBase = aSp & ~((1024*1024)-1);
   126 	if (aStackSize<0 || aStackSize>=1024*1024 || aSp>(TUint)(aStackBase+aStackSize))
   127 		{
   128 		aStackBase=0;	// indicates a daft stack pointer
   129 		return 0;
   130 		}
   131 #else
   132 	// Sanity check & guess at stack size
   133 	aStackBase = aSp & ~((1024*1024)-1);
   134 	aStackSize = (aSp - aStackBase + 4096) & ~4095;		// round up to a multiple of the page size
   135 	if (aStackBase+aStackSize-aSp < 200)
   136 			aStackSize += 4096;		// seems too small - risk another page!
   137 	if (aStackSize<0 || aStackSize>=1024*1024)
   138 		{
   139 		aStackBase=0;	// indicates a daft stack pointer
   140 		return 0;
   141 		}
   142 	aStackSize-=4;	// a clue to the wise that this is a guess...
   143 #endif
   144 
   145 	// Ask the user if we want to risk grabbing the stack...
   146 
   147 	TBuf<0x100> line2;
   148 	line2.Format(KFormatStackInfo, aStackBase, aStackBase+aStackSize-1, aStackSize, aSp);
   149     
   150 	// OK - let stack grabbing commence
   151 	HBufC8* stackbuf = HBufC8::New(aStackSize);
   152 	if (stackbuf==0)
   153 		return 0;
   154 
   155 	TPtr8 stackdes(stackbuf->Des());
   156 	err=RDebug::ReadMemory(aId,aStackBase,stackdes,aStackSize);
   157 	if (err!=KErrNone)
   158 		{
   159 		delete stackbuf;
   160 		stackbuf=0;
   161 		}
   162 	return stackbuf;
   163 	}
   164 
   165 GLDEF_C TInt E32Main()
   166 //
   167 // Reporting more detail of KERN-EXEC 3 errors
   168 //
   169 // Warning: This code uses fragile assumptions which may not be true
   170 // in future releases of EPOC
   171 //
   172 // 1) EXEs are located at 0x20000000
   173 // 2) the map file for an EXE puts the start of text at 400010
   174 // 3) the map file for a DLL puts the start of text at 10000010
   175 // 4) the EPOC ROM lives at address 0x50000000
   176 // 5) EPOC stacks start at a megabyte boundary
   177 //
   178 	{
   179 	TBuf<0x100> line1;
   180 	TBuf<0x100> line2;
   181 	SDebugInfo info;
   182 	struct SRegisterInfo reginfo;
   183 	TUint pc = 0;
   184 	TUint regs[16];
   185 	const TDll* faultDll = NULL;
   186 
   187 	_LIT(KInfo1, "D_EXC started");
   188 	User::InfoPrint(KInfo1);
   189 
   190 	TRequestStatus stat(0);
   191 //	FOREVER
   192 	for (TInt rep=0; rep<2; rep++)	// die after two exceptions
   193 		{
   194 		TInt err = KErrNone;
   195 		// wait for any thread to panic...
   196 
   197 
   198 		err=RDebug::GetException(info,stat);
   199 		if (err!=KErrNone)
   200 			{
   201 			_LIT(KInfo2, "RDebug failure");
   202 			User::Panic(KInfo2, err);
   203 			}
   204 
   205 		User::WaitForRequest(stat);
   206 
   207 		_LIT(KFormatPanic, "%S panic %S %d\r");
   208 		_LIT(KFormatException, "%S exception type %d\r");
   209 		_LIT(KFormatBreakpoint, "%S has breakpoint %d\r");
   210 
   211 
   212 		switch (info.iDebugType)
   213 			{
   214 		case EPanic:
   215 			line1.Format(KFormatPanic,&info.iName,&info.iCategory, info.iPanicType);
   216 			break;
   217 		case EException:
   218 			line1.Format(KFormatException,&info.iName,info.iPanicType);
   219 			break;
   220 		case EBreakPoint:
   221 			line1.Format(KFormatBreakpoint,&info.iName,info.iPanicType);
   222 			break;
   223 		default:
   224 			continue;
   225 			}
   226 
   227 		// assume that it's KERN-EXEC 3 and try to use the 
   228 		// full RDebug support to locate the faulting instruction
   229 
   230 		HBufC8* stack = NULL;
   231 		TInt stackbase=0;
   232 		TInt stacksize=0;
   233 
   234 		const TInt KStackPointerReg = 13;
   235 
   236 		pc = 0x00000001;	// illegal value
   237 
   238 
   239 		err = RDebug::Open();
   240 		if (err==KErrNone)
   241 			{
   242 			err = RDebug::RegisterInfo(reginfo);
   243 			if (!err)
   244 				{
   245 				RDebug::GetRegister(info.iId,reginfo.iNumberOfPcRegister, pc);
   246 				for (int i=0; i<16; i++)
   247 				    {
   248 					RDebug::GetRegister(info.iId, i, regs[i]);
   249 				    }
   250 				}
   251 
   252 			TDllList::FindDlls();
   253 
   254 			stack=GrabStack(line1, info.iId, regs[KStackPointerReg], stackbase, stacksize);
   255 
   256 			RDebug::Close();
   257 			}
   258 
   259 
   260 		_LIT(KFormatEXE,	"pc=%08x, .map equivalent %08x\r");
   261 		_LIT(KFormatROM,	"pc=%08x, in ROM\r");
   262 		_LIT(KFormatDll,	"pc=%08x, in DLL %S, .map equivalent %08x\r");
   263 		_LIT(KFormatOther,	"pc=%08x, iCodeAddr=%08x\r");
   264 		_LIT(KFormatError,	"(Unable to determine pc)\r");
   265 
   266 
   267 		if ((pc&3) == 0)
   268 			{
   269 			if (pc >= 0x20000000 && pc < 0x30000000)
   270 			    {
   271 				line2.Format(KFormatEXE, pc, pc-0x20000000+0x400010);
   272 			    }
   273 			else if (pc >= 0x50000000 && pc < 0x60000000)
   274 				{
   275                 line2.Format(KFormatROM, pc);
   276 				}
   277 			else if (TDllList::Match(pc, faultDll)==KErrNone)
   278 			    {
   279 				line2.Format(KFormatDll, pc, &faultDll->iName, pc-(faultDll->iBase)+0x10000010);
   280 			    }
   281 			else
   282 			    {
   283 				line2.Format(KFormatOther, pc, info.iCodeAddr);
   284 			    }
   285 			}
   286 		else
   287 		    {
   288 			line2.Copy(KFormatError);
   289 		    }
   290 		
   291 		RFs fs;
   292 		err = fs.Connect();
   293 		if (err!=KErrNone)
   294 		    {
   295 			break;
   296 		    }
   297 
   298 		_LIT(KFormatFilename,"d:\\d_exc_%d.txt");
   299 		_LIT(KFormatStackname,"d:\\d_exc_%d.stk");
   300 
   301 		// Report the basic information about registers, DLLs etc
   302 
   303 		TFileName name;
   304 		name.Format(KFormatFilename, *(TUint*)&info.iId);
   305 
   306 		RFile file;
   307 		err=file.Replace(fs, name, EFileWrite+EFileShareAny+EFileStreamText);
   308 		if (err!=KErrNone)
   309 		    {
   310             fs.Close();
   311             break;
   312 		    }
   313 
   314 		TFileText textfile;
   315 		textfile.Set(file);
   316 
   317 		textfile.Write(line1);
   318 		textfile.Write(line2);
   319 		textfile.Write(KBlankLine);
   320 
   321 		_LIT(KFormatRegisters,"r%02d=%08x %08x %08x %08x\r");
   322 
   323 		line2.Format(KFormatRegisters, 0, regs[0], regs[1], regs[2], regs[3]);
   324 		textfile.Write(line2);
   325 		line2.Format(KFormatRegisters, 4, regs[4], regs[5], regs[6], regs[7]);
   326 		textfile.Write(line2);
   327 		line2.Format(KFormatRegisters, 8, regs[8], regs[9], regs[10], regs[11]);
   328 		textfile.Write(line2);
   329 		line2.Format(KFormatRegisters, 12,regs[12], regs[13], regs[14], regs[15]);
   330 		textfile.Write(line2);
   331 
   332 		if (stackbase!=0)
   333 			{
   334 			line2.Format(KFormatStackInfo, stackbase, stackbase+stacksize-1, stacksize, regs[KStackPointerReg]);
   335 			textfile.Write(line2);
   336 			}
   337 
   338 		TDllList::ListDlls(textfile);
   339 		TDllList::ResetAndDestroy();
   340 
   341 		file.Close();
   342 
   343 		// Dump the stack as binary data
   344 
   345 		if (stack)
   346 			{
   347 			name.Format(KFormatStackname, *(TUint*)&info.iId);
   348 			err=file.Replace(fs, name, EFileWrite+EFileShareAny+EFileStreamText);
   349 			if (err==KErrNone)
   350 				{
   351 				file.Write(*stack);
   352 				file.Close();
   353 				}
   354 			}
   355 		delete stack;
   356 
   357 		fs.Close();
   358 		}
   359 	return stat.Int();
   360 	}
   361 
   362