kerneltest/e32utils/analyse/symbols.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Mon, 18 Jan 2010 21:31:10 +0200
changeset 11 329ab0095843
parent 9 96e5fb8b040d
permissions -rw-r--r--
Revision: 201003 Kit: 201003

// Copyright (c) 2000-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:
//

#include "analyse.h"
#include "symbols.h"

#ifdef __MSVCDOTNET__
#include <fstream>
#else //!__MSVCDOTNET__
#include <fstream.h>
#endif //__MSVCDOTNET__

namespace {
unsigned ParseHex(const char* aText, int aLength)
	{
	unsigned value = 0;
	const char* end = aText + aLength;
	do
		{
		unsigned c = *aText++ - '0';
		if (c > 9)
			c -= 'a' - '0' - 10;
		value = (value << 4) + c;
		} while (aText < end);
	return value;
	}
};


// class SymbolFile

SymbolFile::SymbolFile(const char* aSymbolFile, bool aRofs)
	:iText(0), iTextLower(0)
	{
	ifstream file;
#ifdef __MSVCDOTNET__
	file.open(aSymbolFile, ios::binary);
#else //!__MSVCDOTNET__
	file.open(aSymbolFile, ios::nocreate | ios::binary);
#endif //__MSVCDOTNET__
	if (!file)
		{
		cerr << "Unable to open ROM symbol file '" << aSymbolFile << '\'' << endl;
		Analyse::Abort();
		}
//
	file.seekg(0, ios::end);
	iLength = file.tellg();
//
	iText = new char[iLength+1];
	file.seekg(0, ios::beg);
	file.read(iText, iLength);
	iText[iLength] = '\0';
//
	file.close();

	if (aRofs)
		{
		iTextLower = new char[iLength+1];
		for(char *p = iTextLower, *c = iText, *end = iText + iLength;c < end;c++, p++) 
			*p = tolower(*c);
		}
	}

SymbolFile::~SymbolFile()
	{
	delete [] iText;
	if (iTextLower)
		delete [] iTextLower;
	}

bool SymbolFile::Parse(SymbolFile::Parser& aParser, const char* aModuleName, PC aAddress, PC aModuleLength, int aModuleId) const
	{
	char* text = 0;
	PC first_pc_limit = 0, last_pc_limit = 0;
	TState state = EPreFile;
	PC lastPC = 0;
	bool inSyms = false;
	const char *prevName = 0;
	int lastLen = 0;

	if (aModuleName) // should parse only one module 
		{
		bool not_found = false;
		char * name = strstr(iTextLower, aModuleName);
		if (name)
			{
			for(char * p = name; p != iTextLower && *p != '\n';p--);
			if (*p == '\n' || p == iTextLower) 
				{
				name = ++p;
				text = iText + (name - iTextLower);
				}
			else
				not_found = true;
			}
		else
			not_found = true;

		if (not_found)
			return false;

		state = EFile;
		}
	else
		text = iText;
		
	const char* end = iText + iLength;
	while (text < end && state != EError)
		{
		char* endl = strchr(text, '\r');
		if (endl == 0)
			{
			if (!aModuleName)
				{
				state = EError;
				break;
				}
			else
				{
				char* p = text + strlen(text);
				if (*(p+1) == '\n')
					endl = p;
				else
					{
					state = EError;
					break;
					}
				}
			}
		switch (state)
			{
		case EPreFile:
			if (endl != text)
				state = EError;
			else
				state = EFile;
			break;
		case EFile:
			if (strncmp(text, "From", 4) != 0)
				state = EError;
			else
				{
				*endl = '\0';
				char* name = strrchr(text, '\\');
				if (name == 0)
					name = text + 8;
				else
					++name;
				aParser.File(name);
				state = EPostFile;
				}
			break;
		case EPostFile:
			if (endl != text)
				state = EError;
			else
				state = ESymbol;
			break;
		case ESymbol:
			if (text == endl)
				{
				if (aModuleName)
					goto Quit; 
				else
					state = EFile;
				}
			else
				{
				PC pc = ParseHex(text, 8);
				pc &= ~1; // should be odd address, error in symbol table

				if(aModuleName)
					pc += aAddress;

				*endl = '\0';

				char* codeType = strrchr(text+20, '(');

				if ( codeType == NULL || (strcmp(codeType,"(.data)") != 0 &&
					 strcmp(codeType,"(.bss)") != 0 &&
					 strcmp(codeType,"(.init_array)") != 0 &&
					 strcmp(codeType,"(linker$$defined$$symbols)") != 0 &&
					 strcmp(codeType,"(ExportTable)") != 0) )
					{
					if(inSyms && (pc > (lastPC + lastLen)))
						{
						memcpy((void *)(prevName - 15), "<static> after ", 15);
						aParser.Symbol(prevName - 15, lastPC + lastLen, pc - (lastPC + lastLen));
						}

					int length = ParseHex(text + 12, 4);

					if(pc >= lastPC + lastLen)
						{
						bool is_added = aParser.Symbol(text + 20, pc, length);
						if (is_added && aModuleName && !first_pc_limit) 
							{
							first_pc_limit = pc + length;
							last_pc_limit = pc + aModuleLength;
							}
						}

					prevName = text + 20;
					if(pc + length > lastPC + lastLen)
						{
						lastLen = length;
						lastPC = pc;
						}
					inSyms = true;
					}
				}
			break;
		case EError:
			break;
			}
		text = endl + 2;
		}
	if (state == EError)
		Analyse::Abort("Bad ROM symbol file format");

Quit:
	if(aModuleName && lastPC == first_pc_limit && (lastPC + lastLen) < last_pc_limit)
		{
		memcpy((void *)(prevName - 10), "<anon> in ", 10);
		aParser.Symbol(prevName - 10, lastPC + lastLen, last_pc_limit - (lastPC + lastLen));
		}
	aParser.Done(first_pc_limit, last_pc_limit, aModuleId);

	return true;
	}