imgtools/romtools/rombuild/r_areaset.h
author Richard Taylor <richard.i.taylor@nokia.com>
Thu, 22 Apr 2010 11:01:50 +0100
changeset 436 6afacfeb2f3a
parent 0 044383f39525
child 590 360bd6b35136
permissions -rw-r--r--
Added tag 2.12.5 for changeset 7006bcce5299

/*
* 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: 
* Area-related API
*
*/


#ifndef __R_AREASET_H__
#define __R_AREASET_H__

#include <assert.h>
#include <string.h>
#include <stdlib.h>
#include <assert.h>
#include <iostream>
#include <string>
#include <vector>
#include <map>
#include <fstream>

#include <e32std.h>
#include <e32rom.h>				// TLinAddr

typedef std::vector<std::string> TStringList;
struct DepInfo
{
public:
	TBool dependOthers;
	TBool beenDepended;
	int index;
	std::string portName;
	TStringList depFilesList;
	DepInfo()
	{
		dependOthers = EFalse;
		beenDepended = EFalse;
		index = -1;
	}
};
typedef std::map<std::string, DepInfo> TDepInfoList;

class TRomBuilderEntry;

/**
 A zone of memory in which files are stored.

 Areas (except the default one - see below) are relocated from ROM to
 RAM at boot time.

 When created an area is given a "destination base address" (the start
 of the area in RAM) and a maximum size.

 During processing the "source base address" (the start of the area in
 ROM) is set once and the area is extended each time a file is
 processed by setting repeatedly the "source limit address" (the top
 of the area in ROM).

 The default area contains files that don't need relocation.  So its
 "source base address" and "destination base address" are the same.

 @private 
*/

class Area
	{
public:
	~Area();

	const char* Name() const;

	void SetSrcBaseAddr(TLinAddr aSrcBaseAddr);
	TLinAddr SrcBaseAddr() const;

	TBool ExtendSrcLimitAddr(TLinAddr aSrcLimitAddr, TUint& aOverflow);
	TLinAddr SrcLimitAddr() const;
	
	TLinAddr DestBaseAddr() const;

	TUint MaxSize() const;
	TUint UsedSize() const;

	TBool IsDefault() const;

	void AddFile(TRomBuilderEntry* aFile);

	TInt SortFilesForPagedRom();
private:
	// only AreaSet can create areas
	Area(const char* aName, TLinAddr aDestBaseAddr, TUint aMaxSize, Area* aNext=0);
	void ReleaseAllFiles();
	void WriteDependenceGraph();
public:
	TRomBuilderEntry* iFirstPagedCode; // For PagedRom only
private:
	const char* iName;
	TLinAddr iDestBaseAddr;
	TLinAddr iSrcBaseAddr;
	TLinAddr iSrcLimitAddr;
	TUint iMaxSize;

	TBool iIsDefault;

	TRomBuilderEntry* iFiles;
	TRomBuilderEntry** iNextFilePtrPtr;

	Area* iNextArea;

	friend class AreaSet;
	friend class FilesInAreaIterator;
	friend class NonDefaultAreasIterator;
	};


inline  const char* Area::Name() const
	{
	return iName;
	}


inline void Area::SetSrcBaseAddr(TLinAddr aSrcBaseAddr)
	{
	// setting allowed only once
	assert(iSrcBaseAddr == 0);	
	assert(aSrcBaseAddr != 0);

	iSrcLimitAddr = iSrcBaseAddr = aSrcBaseAddr;
	}


inline TLinAddr Area::SrcBaseAddr() const
	{
	// must have been set before
	assert(iSrcBaseAddr != 0);
	return iSrcBaseAddr;
	}


inline TLinAddr Area::SrcLimitAddr() const
	{
	// must have been set before
	assert(iSrcBaseAddr != 0);
	return iSrcLimitAddr;
	}


inline TLinAddr Area::DestBaseAddr() const
	{
	return iDestBaseAddr;
	}


inline TUint Area::MaxSize() const
	{
	return iMaxSize;
	}


inline TUint Area::UsedSize() const
	{
	return iSrcLimitAddr-iSrcBaseAddr;
	}


inline TBool Area::IsDefault() const
	{
	return iIsDefault;
	}


////////////////////////////////////////////////////////////////////////

class TRomBuilderEntry;

/**
 Iterate over every file in a given area.

 Files are iterated in the order in which they have been appended to
 the area.  
 
 @private
*/

class FilesInAreaIterator
	{
public:
	FilesInAreaIterator(const Area& aArea);

	TBool IsDone() const;
	TRomBuilderEntry* Current() const;
	void GoToNext();

private:
	TRomBuilderEntry* iCurrentFile;
	};


inline FilesInAreaIterator::FilesInAreaIterator(const Area& aArea)
	: iCurrentFile(aArea.iFiles)
	{
	}

inline TBool FilesInAreaIterator::IsDone() const
	{
	return iCurrentFile == 0;
	}

inline TRomBuilderEntry* FilesInAreaIterator::Current() const
	{
	return iCurrentFile;
	}

////////////////////////////////////////////////////////////////////////

/**
 Set of areas indexed by name.

 There can be only one default area identified by its name
 (KDefaultAreaName).

 @private 
*/

class AreaSet
	{
public:
	enum TAddResult
		{ 
		EAdded,
		EOverlap,
		EDuplicateName,
		EOverflow,
		};

public:
	AreaSet();
	~AreaSet();

	TAddResult AddArea(const char* aName, TLinAddr aDestBaseAddr, TUint aMaxSize, const char*& aOverlappingArea);
	void ReleaseAllAreas();

	Area* FindByName(const char* aName) const;
	TInt Count() const;
	Area* DefaultArea() const;

private:
	Area* iNonDefaultAreas;
	Area* iDefaultArea;
	TInt iAreaCount;

public:
	static const char KDefaultAreaName[];

	friend class NonDefaultAreasIterator;
	};


inline TInt AreaSet::Count() const
	{
	return iAreaCount;
	}


inline Area* AreaSet::DefaultArea() const
	{
	return iDefaultArea;
	}


////////////////////////////////////////////////////////////////////////


/**

 Iterate over every non-default area of a given area set.

 @private
*/

class NonDefaultAreasIterator
	{
public:
	NonDefaultAreasIterator(const AreaSet& aAreaSet);

	TBool IsDone() const;
	Area& Current() const;
	void GoToNext();

private:
	Area* iCurrentArea;
	};

inline NonDefaultAreasIterator::NonDefaultAreasIterator(const AreaSet& aAreaSet)
	: iCurrentArea(aAreaSet.iNonDefaultAreas)
	{
	}

inline TBool NonDefaultAreasIterator::IsDone() const
	{
	return iCurrentArea == 0;
	}

inline Area& NonDefaultAreasIterator::Current() const
	{
	assert(iCurrentArea != 0);
	return *iCurrentArea;
	}

#endif // __R_AREASET_H__