libraries/ltkutils/src/mapfile.cpp
author Tom Sutcliffe <thomas.sutcliffe@accenture.com>
Wed, 23 Jun 2010 15:52:26 +0100
changeset 0 7f656887cf89
permissions -rw-r--r--
First submission to Symbian Foundation staging server.

// mapfile.cpp
// 
// Copyright (c) 2010 Accenture. All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the "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:
// Accenture - Initial contribution
//
#include <fshell/bsym.h>
#include <fshell/ltkutils.h>
#include "bsymtree.h"
#include <fshell/descriptorutils.h>

#include <fshell/iocli.h>
using namespace LtkUtils;
using namespace IoUtils;

const TInt KSymbolGranularity = 256; // I have no idea...

struct SSymbol
	{
	TUint iAddress;
	TUint iLength;
	HBufC8* iName;
	};

NONSHARABLE_CLASS(CMapFileImpl) : public CMapFile
	{
public:
	CMapFileImpl();
	~CMapFileImpl();
	void ConstructL(RFs& aFs, const TDesC& aFileName);
	void DoLookup(TUint32 aOffsetInCodeSeg, TDes& aResult);
	RNode* DoCreateCompletionTreeL();

private:
	RArray<SSymbol> iSymbols;
	};

EXPORT_C CMapFile* CMapFile::NewL(RFs& aFs, const TDesC& aFileName)
	{
	CMapFileImpl* result = new (ELeave) CMapFileImpl;
	CleanupStack::PushL(result);
	//CleanupStack::PushL((CBase*)1); //DEBUG
	result->ConstructL(aFs, aFileName);
	//CleanupStack::Pop(); //DEBUG
	CleanupStack::Pop(result);
	return result;
	}

CMapFile::CMapFile()
	{
	}

CMapFileImpl::CMapFileImpl()
	: iSymbols(KSymbolGranularity, _FOFF(SSymbol, iAddress))
	{
	}

CMapFileImpl::~CMapFileImpl()
	{
	const TInt count = iSymbols.Count();
	for (TInt i = 0; i < count; i++)
		{
		delete iSymbols[i].iName;
		}
	iSymbols.Close();
	}

EXPORT_C CMapFile::~CMapFile()
	{
	iReadBuf.Close();
	delete iFileName;
	}

#undef LeaveIfErr
#define LeaveIfErr(args...) StaticLeaveIfErr(args)

void CMapFileImpl::ConstructL(RFs& aFs, const TDesC& aFileName)
	{
	iFileName = aFileName.AllocL();
	iReadBuf.CreateL(4096);
	LeaveIfErr(iFile.Open(aFs, aFileName, EFileShareReadersOnly), _L("Couldn't open file %S"), &aFileName);

	TPtrC8 line;
	TBool rvct = EFalse;
	_LIT8(KRvct, "ARM Linker, RVCT");
	if (GetNextLine(line) && HasPrefix(line, KRvct)) rvct = ETrue;
	TBool foundOffset = EFalse;
	TInt linenum = 0; // for debugging

	if (rvct)
		{
		TInt local = ETrue;
		while (GetNextLine(line))
			{
			linenum++;
			if (line.Length() == 0) continue;
			_LIT8(KRVCTOffsetLine, "    Image$$ER_RO$$Base                       ");
			if (!foundOffset)
				{
				if (HasPrefix(line, KRVCTOffsetLine))
					{
					TLex8 lex(line.Mid(KRVCTOffsetLine().Length()));
					iTextOffset = HexLexL(lex);
					foundOffset = ETrue;

					// No go through any symbols we found before this, and correct their offsets
					for (TInt i = 0; i < iSymbols.Count(); i++)
						{
						iSymbols[i].iAddress -= iTextOffset;
						}
					continue;
					}
				}

			_LIT8(KGlobalSection, "    Global Symbols");
			if (HasPrefix(line, KGlobalSection)) local = EFalse; // We've reached the global section, don't have to ignore so much stuff

			SSymbol symbol;

			// This is rather awkward, because there's no proper delimiting between symbol name and the next stuff
			_LIT8(KSpaces, "   ");
			if (line.Length() < 55) continue;
			TInt foundSpaces = line.Mid(55).Find(KSpaces);
			if (foundSpaces == KErrNotFound) continue;
			TInt spp = 55 + foundSpaces - 10;
			symbol.iName = line.Left(spp).AllocLC();
			symbol.iName->Des().Trim();
				
			TLex8 lex(line);
			lex.Inc(spp);
			symbol.iAddress = HexLexL(lex) - iTextOffset;
			lex.SkipSpace();
			if (local)
				{
				// In local symbols section we have to worry about non-code stuff
				_LIT8(KArm,   "ARM Code  ");
				_LIT8(KThumb, "Thumb Code");
				TPtrC8 type = lex.Remainder().Left(KThumb().Length());
				if (type != KArm && type != KThumb)
					{
					CleanupStack::PopAndDestroy(symbol.iName);
					continue;
					}
				}

			lex.Inc(10); // Far enough to get over "ARM Code" or "Thumb Code"
			lex.SkipSpace();
			User::LeaveIfError(lex.Val(symbol.iLength));
			if (symbol.iLength == 0)
				{
				CleanupStack::PopAndDestroy(symbol.iName);
				continue; // todo thunks lie about their size...
				}

			User::LeaveIfError(iSymbols.Append(symbol));
			CleanupStack::Pop(symbol.iName);
			}
		}
	else
		{
		while (GetNextLine(line))
			{
			linenum++;
			_LIT8(KGccTextOffsetLine, "Address of section .text set to ");
			if (HasPrefix(line, KGccTextOffsetLine))
				{
				// GCCE style
				TLex8 lex(line.Mid(KGccTextOffsetLine().Length()));
				iTextOffset = HexLexL(lex);
				continue;
				}

			_LIT8(KFin, ".fini");
			if (line == KFin) break;
			if (line.Length() < 16) continue;
			TLex8 lex(line);
			lex.Inc(16);
			SSymbol symbol;
			symbol.iLength = 4; // unless otherwise updated by the next symbol addr
			TInt err = HexLex(lex, symbol.iAddress);
			if (err) continue;
			symbol.iAddress -= iTextOffset;
			lex.SkipSpace();
			if (lex.Offset() < 42) continue; // Code symbols have space up to column 42
			symbol.iName = lex.Remainder().AllocLC();

			User::LeaveIfError(iSymbols.Append(symbol));
			CleanupStack::Pop(symbol.iName);
			}
		// GCCE doesn't even sort the symbols in its map file. Unbelievable! Or possibly even inconceivable!
		iSymbols.SortUnsigned();
		for (TInt i = 1; i < iSymbols.Count(); i++)
			{
			SSymbol& prevSymbol = iSymbols[i - 1];
			prevSymbol.iLength = iSymbols[i].iAddress - prevSymbol.iAddress;
			}
		}
	iFile.Close();
	iReadBuf.Close();
	}

TBool CMapFile::GetNextLine(TPtrC8& aPtr)
	{
	_LIT8(KNewline, "\r\n");
	iReadBuf.Delete(0, aPtr.Length());
	if (HasPrefix(iReadBuf, KNewline)) iReadBuf.Delete(0, KNewline().Length());

	TInt newline = iReadBuf.Find(KNewline);
	if (newline != KErrNotFound)
		{
		aPtr.Set(iReadBuf.Left(newline));
		return ETrue;
		}
	// Otherwise need to try reading some more from file
	TPtr8 restOfDesc((TUint8*)iReadBuf.Ptr() + iReadBuf.Size(), 0, iReadBuf.MaxSize() - iReadBuf.Size());
	TInt err = iFile.Read(restOfDesc);
	if (err) restOfDesc.Zero();
	iReadBuf.SetLength(iReadBuf.Length() + restOfDesc.Length());
	
	// Now re-try looking for newline
	newline = iReadBuf.Find(KNewline);
	if (newline != KErrNotFound)
		{
		aPtr.Set(iReadBuf.Left(newline + KNewline().Length()));
		return ETrue;
		}

	// No more newlines, just return whatever's left in the buffer
	aPtr.Set(iReadBuf);
	return aPtr.Length() != 0; // And return whether it's worth having
	}

EXPORT_C void CMapFile::Lookup(TUint32 aOffsetInCodeSeg, TDes& aResult)
	{
	//TUint32 offset = aOffsetInCodeSeg & 0xFFFFFFFE; // Mask bottom bit as it just indicates thumb mode
	CMapFileImpl* impl = static_cast<CMapFileImpl*>(this); // We know we're actually a CMapFileImpl
	impl->DoLookup(aOffsetInCodeSeg, aResult);
	}

void CMapFileImpl::DoLookup(TUint32 aOffsetInCodeSeg, TDes& aResult)
	{
	TInt pos = 0;
	SSymbol dummy; dummy.iAddress = aOffsetInCodeSeg;
	TBool found = iSymbols.FindInUnsignedKeyOrder(dummy, pos) == KErrNone;
	if (!found && pos != 0) pos--;

	aResult.Zero();
	const SSymbol& symbol = iSymbols[pos];
	if (aOffsetInCodeSeg >= symbol.iAddress && aOffsetInCodeSeg < symbol.iAddress + symbol.iLength)
		{
		aResult.Copy(*symbol.iName);
		aResult.AppendFormat(_L(" + 0x%x"), aOffsetInCodeSeg - symbol.iAddress);
		}
	}

EXPORT_C void CMapFile::GetFileNameL(TDes& aFileName) const
	{
	aFileName.Copy(*iFileName);
	}

RNode* CMapFile::CreateCompletionTreeL()
	{
	CMapFileImpl* impl = static_cast<CMapFileImpl*>(this); // We know we're actually a CMapFileImpl
	return impl->DoCreateCompletionTreeL();
	}

RNode* CMapFileImpl::DoCreateCompletionTreeL()
	{
	RNode* result = RNode::NewL();
	CleanupDeletePushL(result);

	RLtkBuf tempBuf;
	tempBuf.CreateLC(256);
	for (TInt i = 0; i < iSymbols.Count(); i++)
		{
		tempBuf.Zero();
		tempBuf.AppendL(*iSymbols[i].iName);
		tempBuf.ReserveExtraL(1);
		result->InsertStringL(tempBuf.PtrZ(), iSymbols[i].iAddress);
		}
	CleanupStack::PopAndDestroy(&tempBuf);
	CleanupStack::Pop(result);
	return result;
	}