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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
20 _LIT(KBlankLine, "\r");
25 TDll(const TDesC& aName, TUint8* aBase, TInt aSize)
26 : iName(aName), iBase((TUint)aBase), iSize((TUint)aSize)
37 static void FindDlls();
38 static void ResetAndDestroy();
39 static void ListDlls(TFileText& aTextFile);
40 static TInt Match(TUint anAddr, const TDll*& aDll);
43 static RPointerArray<TDll> iList;
46 RPointerArray<TDll> TDllList::iList(10);
48 void TDllList::ResetAndDestroy()
50 iList.ResetAndDestroy();
53 void TDllList::ListDlls(TFileText& aTextFile)
57 _LIT(KDllFormat, "%08x-%08x %S\r");
59 aTextFile.Write(KBlankLine);
60 for (TInt i=0; i<iList.Count(); i++)
62 const TDll& dll=*(iList[i]);
63 line.Format(KDllFormat, dll.iBase, dll.iBase+dll.iSize-1, &dll.iName);
64 aTextFile.Write(line);
68 TInt TDllList::Match(TUint anAddr, const TDll*& aDll)
70 for (TInt i=0; i<iList.Count(); i++)
73 if (aDll->iBase<=anAddr && aDll->iBase+aDll->iSize > anAddr)
80 void TDllList::FindDlls()
82 // For each library known to the system, look for a chunk of the same name
88 while (findLib.Next(libName)==KErrNone)
90 TFindChunk findChunk(libName);
92 if (findChunk.Next(chunkName)!=KErrNone)
95 if (chunk.Open(findChunk)!=KErrNone)
97 TUint8* base=chunk.Base();
98 TInt size=chunk.Size();
100 TDll* dllptr=new TDll(libName,base,size);
102 iList.Append(dllptr);
106 _LIT(KFormatStackInfo,"Stack %08x-%08x (? %d?), sp=%08x\r");
108 HBufC8* GrabStack(const TDesC& /*aLine1*/, TThreadId aId, TUint aSp, TInt& aStackBase, TInt& aStackSize)
112 // Defect in E32 162 which means that RThread::GetRamSizes will always reset the machine!
114 if (thread.Open(aId)!=KErrNone)
118 err=thread.GetRamSizes(heapsize,aStackSize);
125 aStackBase = aSp & ~((1024*1024)-1);
126 if (aStackSize<0 || aStackSize>=1024*1024 || aSp>(TUint)(aStackBase+aStackSize))
128 aStackBase=0; // indicates a daft stack pointer
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)
139 aStackBase=0; // indicates a daft stack pointer
142 aStackSize-=4; // a clue to the wise that this is a guess...
145 // Ask the user if we want to risk grabbing the stack...
148 line2.Format(KFormatStackInfo, aStackBase, aStackBase+aStackSize-1, aStackSize, aSp);
150 // OK - let stack grabbing commence
151 HBufC8* stackbuf = HBufC8::New(aStackSize);
155 TPtr8 stackdes(stackbuf->Des());
156 err=RDebug::ReadMemory(aId,aStackBase,stackdes,aStackSize);
165 GLDEF_C TInt E32Main()
167 // Reporting more detail of KERN-EXEC 3 errors
169 // Warning: This code uses fragile assumptions which may not be true
170 // in future releases of EPOC
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
182 struct SRegisterInfo reginfo;
185 const TDll* faultDll = NULL;
187 _LIT(KInfo1, "D_EXC started");
188 User::InfoPrint(KInfo1);
190 TRequestStatus stat(0);
192 for (TInt rep=0; rep<2; rep++) // die after two exceptions
195 // wait for any thread to panic...
198 err=RDebug::GetException(info,stat);
201 _LIT(KInfo2, "RDebug failure");
202 User::Panic(KInfo2, err);
205 User::WaitForRequest(stat);
207 _LIT(KFormatPanic, "%S panic %S %d\r");
208 _LIT(KFormatException, "%S exception type %d\r");
209 _LIT(KFormatBreakpoint, "%S has breakpoint %d\r");
212 switch (info.iDebugType)
215 line1.Format(KFormatPanic,&info.iName,&info.iCategory, info.iPanicType);
218 line1.Format(KFormatException,&info.iName,info.iPanicType);
221 line1.Format(KFormatBreakpoint,&info.iName,info.iPanicType);
227 // assume that it's KERN-EXEC 3 and try to use the
228 // full RDebug support to locate the faulting instruction
230 HBufC8* stack = NULL;
234 const TInt KStackPointerReg = 13;
236 pc = 0x00000001; // illegal value
239 err = RDebug::Open();
242 err = RDebug::RegisterInfo(reginfo);
245 RDebug::GetRegister(info.iId,reginfo.iNumberOfPcRegister, pc);
246 for (int i=0; i<16; i++)
248 RDebug::GetRegister(info.iId, i, regs[i]);
252 TDllList::FindDlls();
254 stack=GrabStack(line1, info.iId, regs[KStackPointerReg], stackbase, stacksize);
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");
269 if (pc >= 0x20000000 && pc < 0x30000000)
271 line2.Format(KFormatEXE, pc, pc-0x20000000+0x400010);
273 else if (pc >= 0x50000000 && pc < 0x60000000)
275 line2.Format(KFormatROM, pc);
277 else if (TDllList::Match(pc, faultDll)==KErrNone)
279 line2.Format(KFormatDll, pc, &faultDll->iName, pc-(faultDll->iBase)+0x10000010);
283 line2.Format(KFormatOther, pc, info.iCodeAddr);
288 line2.Copy(KFormatError);
298 _LIT(KFormatFilename,"d:\\d_exc_%d.txt");
299 _LIT(KFormatStackname,"d:\\d_exc_%d.stk");
301 // Report the basic information about registers, DLLs etc
304 name.Format(KFormatFilename, *(TUint*)&info.iId);
307 err=file.Replace(fs, name, EFileWrite+EFileShareAny+EFileStreamText);
317 textfile.Write(line1);
318 textfile.Write(line2);
319 textfile.Write(KBlankLine);
321 _LIT(KFormatRegisters,"r%02d=%08x %08x %08x %08x\r");
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);
334 line2.Format(KFormatStackInfo, stackbase, stackbase+stacksize-1, stacksize, regs[KStackPointerReg]);
335 textfile.Write(line2);
338 TDllList::ListDlls(textfile);
339 TDllList::ResetAndDestroy();
343 // Dump the stack as binary data
347 name.Format(KFormatStackname, *(TUint*)&info.iId);
348 err=file.Replace(fs, name, EFileWrite+EFileShareAny+EFileStreamText);