lafagnosticuifoundation/cone/inc/CoeDynamicStorage.inl
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Mon, 15 Mar 2010 12:41:34 +0200
branchRCL_3
changeset 10 9f56a4e1b8ab
parent 0 2f259fa3e83a
permissions -rw-r--r--
Revision: 201009 Kit: 201010

// Copyright (c) 1997-2010 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "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:
//

// #define PROFILE_MEMORY	// Uncomment to profile ammount of saved memory.

#ifdef PROFILE_MEMORY
static TInt MEMORY_SAVED = 0;	// In bytes
const TInt KCurrentNumberOfOptionalVaraiblesInCCoeControl = 13;
#endif

/** 
This class implements a container for (in most cases, pointers to) optional 
CCoeControl members.

The size of the class is 12 bytes: one 32-bit pointer to a C-style array 
of 32-bit data, and one long long (64 bits) variable used as an array of 4-bit 
indices into said C-style array. 

Each optional CCoeControl data is assigned an index into the long long member, 
used to look up the variables actual index in the C-style array. This way, the 
first time a new optional CCoeControl variable is set it is appended to the C-style 
array, and the index of the new item is recorded in the variable's assigned 
4-bit slot in the long long. 

The 4-bits used for each index allows for 16 indices to be stored in the long long. 
However, one value (say 0xF) is required to signal that the optional variable has 
not yet been added to the C-style array, so this leaves 15 optional CCoeControl members. 

This also leaves the last 4 bits in the long long unused. Now, we also need a way 
to record the current size of the C-style array. As the size range from 0 to 15, 
4-bit is enough, which means that we can use the last 4 spare bits in the long long 
instead of adding another variable to the RCoeDynamicDataStorage!

Thus, each first time an optional member is set, we need to re-allocate the C-style 
array, and this operation might of course fail. All of the data kept in the 
RCoeDynamicDataStorage must therefore be optional, and the code able to cope 
with this. To minimize the number of re-allocations, the RCoeDynamicDataStorage 
allows pre-allocation of the maximum amount needed. Currently, pre-allocation is 
attempted at CCoeControl construction (in the c'tor), and compression is attempted 
when the control is activated. The RCoeDynamicDataStorage will still work if either 
fails.

@internalComponent */
class RCoeDynamicDataStorage
	{
public:
	RCoeDynamicDataStorage();
	TInt Open();
	void Close();
	TInt Count() const;
	void ResetCount();
	void IncCount();
	TInt ReserveData();
	TBool HasReservedData() const;
	void AttemptCompress();
	inline TAny* Data(TInt aSlotIndex, TAny* aDefaultValue) const;	//mm Will this method be expanded inline when used inside coecntrl.cpp?
	TInt SetData(TInt aSlotIndex, TAny* aAny);
	inline TInt DataIndex(TInt aSlotIndex) const;
	void SetDataIndex(TInt aSlotIndex, TInt aDataIndex);
	
private:
	static const TInt KMaxDataItems;
	static const TInt KCounterSlotIndex;
	static const TInt KIndexSlotSize;
	static const Uint64 KIndexSlotMask;
	static const TInt KUndefinedIndex;

private:
	TInt Append(TAny* aAny);
	
private:	
	Uint64 iIndices;	// array of slots containing indices into the data array
	TAny** iDynamicData;		// dynamic array of 32-bit data (mostly pointers)
	};
	

const TInt RCoeDynamicDataStorage::KMaxDataItems = 15;
const TInt RCoeDynamicDataStorage::KCounterSlotIndex = 0xF;
const TInt RCoeDynamicDataStorage::KIndexSlotSize = 4;
const Uint64 RCoeDynamicDataStorage::KIndexSlotMask = 0xFLL;	//long long
const TInt RCoeDynamicDataStorage::KUndefinedIndex = 0xF;

RCoeDynamicDataStorage::RCoeDynamicDataStorage()
: iIndices(~0x0), iDynamicData(NULL)	// Set iIndices to all ones
	{
	ResetCount();
	}

TInt RCoeDynamicDataStorage::Open()
	{
	ASSERT(!iDynamicData);
	TInt err = ReserveData();

#ifdef PROFILE_MEMORY
	if (err==KErrNone)
		{
		MEMORY_SAVED += KCurrentNumberOfOptionalVaraiblesInCCoeControl*sizeof(TAny*);
		MEMORY_SAVED -= sizeof(RCoeDynamicDataStorage);
		}
#endif

	return err;
	}

void RCoeDynamicDataStorage::Close()
	{
#ifdef PROFILE_MEMORY
	MEMORY_SAVED += sizeof(RCoeDynamicDataStorage);
	if(HasReservedData())
		MEMORY_SAVED +=  KMaxDataItems*sizeof(TAny*);
	else
		MEMORY_SAVED += Count()*sizeof(TAny*);
	
	MEMORY_SAVED -= KCurrentNumberOfOptionalVaraiblesInCCoeControl*sizeof(TAny*);
#endif	
	delete [] iDynamicData;
	}

TInt RCoeDynamicDataStorage::Count() const
	{
	if(HasReservedData())
		{
		TInt count = 0;
		for(TInt i = 0; i < KMaxDataItems; i++)
			{
			if(DataIndex(i) != KUndefinedIndex)
				count++;
			}
		return count;	
		}
	else
		return DataIndex(KCounterSlotIndex);	
	}

void RCoeDynamicDataStorage::ResetCount()
	{
	SetDataIndex(KCounterSlotIndex, 0);
	}
	
void RCoeDynamicDataStorage::IncCount()
	{
	TInt count = Count();
	count++;
	SetDataIndex(KCounterSlotIndex, count);
	ASSERT(count <= KMaxDataItems);
	}

TInt RCoeDynamicDataStorage::SetData(TInt aSlotIndex, TAny* aAny)
	{
	if(!this)
		return KErrNoMemory;
	
	const TInt dataIndex = DataIndex(aSlotIndex);

	if(dataIndex != KUndefinedIndex)
		iDynamicData[dataIndex] = aAny;
	else if(HasReservedData())
		{
		const TInt count = Count();
		iDynamicData[count] = aAny;
		SetDataIndex(aSlotIndex, count);
		}
	else
		{
		const TInt err = Append(aAny);
		if(err)
			return err;
		else
			SetDataIndex(aSlotIndex, Count()-1);
		}
	ASSERT(Data(aSlotIndex, NULL) == aAny);	
	return KErrNone;
	}

inline TAny* RCoeDynamicDataStorage::Data(TInt aSlotIndex, TAny* aDefaultValue) const
	{
	if (!this)
		return aDefaultValue;

	const TInt dataIndex = DataIndex(aSlotIndex);
	if (dataIndex!=KUndefinedIndex && iDynamicData!=NULL)	//Defensive Code
		return iDynamicData[dataIndex];
	else
		return aDefaultValue;
	}

inline TInt RCoeDynamicDataStorage::DataIndex(TInt aSlotIndex) const
	{
	const Uint64 mask = KIndexSlotMask << KIndexSlotSize*aSlotIndex;
	return ((iIndices&mask) >> KIndexSlotSize*aSlotIndex);
	}
	
void RCoeDynamicDataStorage::SetDataIndex(TInt aSlotIndex, TInt aDataIndex)
	{
	const Uint64 mask = KIndexSlotMask << KIndexSlotSize*aSlotIndex;
	const Uint64 dataIndex = Uint64(aDataIndex) << KIndexSlotSize*aSlotIndex;
	iIndices &= ~mask;	// clear the old index
	iIndices |= dataIndex; // set the new index
	}

TInt RCoeDynamicDataStorage::Append(TAny* aAny)
	{
	ASSERT(!HasReservedData());	// Should never have reserved data if we get here
	
	const TInt count = Count();
	TAny** newData = new TAny*[count+1];
	if(!newData)
		return KErrNoMemory;
	
	memcpy(newData, iDynamicData, count * sizeof(TAny*));
		
	newData[count]	= aAny;
	IncCount();
#ifdef PROFILE_MEMORY
	MEMORY_SAVED -= (1 * sizeof(TAny*));
#endif
		
	delete [] iDynamicData;
	iDynamicData = newData;
	return KErrNone;
	}


TInt RCoeDynamicDataStorage::ReserveData()
	{
	ASSERT(!iDynamicData);
	iDynamicData = new TAny*[KMaxDataItems];	// Attempt to pre-alloc data
	if(!iDynamicData)
		return KErrNoMemory;	
	
#ifdef PROFILE_MEMORY
	MEMORY_SAVED -= KMaxDataItems*sizeof(TAny*);
#endif	
	return KErrNone;
	}
		
TBool RCoeDynamicDataStorage::HasReservedData() const
	{
	return (iDynamicData && DataIndex(KCounterSlotIndex) == 0);
	}

void RCoeDynamicDataStorage::AttemptCompress()
	{
	if(!HasReservedData()) 
		return;
	
	const TInt count = Count();
	if(!count)	// "new TAny*[0]" gives undefined result!
		{
		delete [] iDynamicData;
		iDynamicData = NULL;
		}
	else
		{
		TAny** newData = new TAny*[count];
		if(!newData)
			return;
		
		memcpy(newData, iDynamicData, count * sizeof(TAny*));
					
		delete [] iDynamicData;
		iDynamicData = newData;
		SetDataIndex(KCounterSlotIndex, count);
		}
		
#ifdef PROFILE_MEMORY		
	MEMORY_SAVED += (sizeof(TAny*) * (0xF - count));
#endif	
	}

// 
// class TCoeZoomWithType 
// 

class TCoeZoomWithType 
    { 
public:                 
    TInt iZoomFactor; 
    CCoeControl::TZoomType iZoomType; 
    }; 

// 
// class SColorOverride
// 

struct SColorOverride	
	{
	TRgb iColor;
	TInt iLogicalColor;
	};

enum TData
	{
	ECoeControlParent = 0x0,
	ECoeControlObserver = 0x1,
	ECoeComponentArray = 0x2,
	ECoeLayoutManager = 0x3,
	ECoeControlBackground = 0x4,
	EMaximumWidthTInt = 0x5,
	EUniqueHandleTInt = 0x6,
	EColorOverrides = 0x7,
	EFocusObserverNotificationIdentifier = 0x8,
	ECoeControlHitTest = 0x9,
	EZoomWithType = 0xA,
	ECustomGc = 0xB,
	EFontProvider = 0xC,
	
	EMaxOptionlData = 0xF
	};

// CoeControlParent
inline CCoeControl* DoGetParent(const RCoeDynamicDataStorage& aData)
	{ return static_cast<CCoeControl*>(aData.Data(ECoeControlParent, NULL)); }

inline TInt DoSetParent(RCoeDynamicDataStorage& aData, CCoeControl* aParent) 
	{ return aData.SetData(ECoeControlParent, aParent); }

// Observer
inline MCoeControlObserver* DoGetObserver(const RCoeDynamicDataStorage& aData)
	{ return static_cast<MCoeControlObserver*>(aData.Data(ECoeControlObserver, NULL)); }

inline TInt DoSetObserver(RCoeDynamicDataStorage& aData, MCoeControlObserver* aObserver) 
	{ return aData.SetData(ECoeControlObserver, aObserver); }

// ComponentArray
inline CCoeControlArray* DoGetComponentArray(const RCoeDynamicDataStorage& aData)
	{ return static_cast<CCoeControlArray*>(aData.Data(ECoeComponentArray, NULL)); }

inline TInt DoSetComponentArray(RCoeDynamicDataStorage& aData, CCoeControlArray* aComponentArray) 
	{ return aData.SetData(ECoeComponentArray, aComponentArray); }

// LayoutManager
inline MCoeLayoutManager* DoGetLayoutManager(const RCoeDynamicDataStorage& aData)
	{ return static_cast<MCoeLayoutManager*>(aData.Data(ECoeLayoutManager, NULL)); }

inline TInt DoSetLayoutManager(RCoeDynamicDataStorage& aData, MCoeLayoutManager* aLayoutManager) 
	{ return aData.SetData(ECoeLayoutManager, aLayoutManager); }

// Background
inline const MCoeControlBackground* DoGetBackground(const RCoeDynamicDataStorage& aData)
	{ return reinterpret_cast<const MCoeControlBackground*>(aData.Data(ECoeControlBackground, NULL)); }

inline TInt DoSetBackground(RCoeDynamicDataStorage& aData, const MCoeControlBackground* aBackground) 
	{ return aData.SetData(ECoeControlBackground, const_cast<MCoeControlBackground*>(aBackground)); }

// Max width
inline TInt DoGetMaximumWidth(const RCoeDynamicDataStorage& aData)
	{ return reinterpret_cast<TInt>(aData.Data(EMaximumWidthTInt, NULL)); }

inline TInt DoSetMaximumWidth(RCoeDynamicDataStorage& aData, TInt aMaxWidth) 
	{ return aData.SetData(EMaximumWidthTInt, reinterpret_cast<TAny*>(aMaxWidth)); }

// Unique Id
inline TInt DoGetUniqueHandle(const RCoeDynamicDataStorage& aData)
	{ return reinterpret_cast<TInt>(aData.Data(EUniqueHandleTInt, reinterpret_cast<TAny*>(KErrNotFound))); }

inline TInt DoSetUniqueHandle(RCoeDynamicDataStorage& aData, TInt aUniqueHandle) 
	{ return aData.SetData(EUniqueHandleTInt, reinterpret_cast<TAny*>(aUniqueHandle)); }

// ColorOverrides
inline CArrayFix<SColorOverride>* DoGetColorOverrides(const RCoeDynamicDataStorage& aData)
	{ return reinterpret_cast<CArrayFix<SColorOverride>*>(aData.Data(EColorOverrides, NULL)); }

inline TInt DoSetColorOverrides(RCoeDynamicDataStorage& aData, CArrayFix<SColorOverride>* aColorOverrides) 
	{ return aData.SetData(EColorOverrides, static_cast<TAny*>(aColorOverrides)); }

// FocusObserverNotificationIdentifier
inline TInt DoGetFocusObserverNotificationIdentifier(const RCoeDynamicDataStorage& aData)
	{ return reinterpret_cast<TInt>(aData.Data(EFocusObserverNotificationIdentifier, NULL)); }

inline TInt DoSetFocusObserverNotificationIdentifier(RCoeDynamicDataStorage& aData, TInt aFocusObserverNotificationIdentifier) 
	{ return aData.SetData(EFocusObserverNotificationIdentifier, reinterpret_cast<TAny*>(aFocusObserverNotificationIdentifier)); }

// HitTest
inline const MCoeControlHitTest* DoGetHitTest(const RCoeDynamicDataStorage& aData)
	{ return reinterpret_cast<const MCoeControlHitTest*>(aData.Data(ECoeControlHitTest, NULL)); }

inline TInt DoSetHitTest(RCoeDynamicDataStorage& aData, const MCoeControlHitTest* aHitTest) 
	{ return aData.SetData(ECoeControlHitTest, const_cast<MCoeControlHitTest*>(aHitTest)); }

// ZoomWithType
inline TCoeZoomWithType* DoGetZoomWithType(const RCoeDynamicDataStorage& aData)
	{ return reinterpret_cast<TCoeZoomWithType*>(aData.Data(EZoomWithType, NULL)); }

inline TInt DoSetZoomWithType(RCoeDynamicDataStorage& aData, const TCoeZoomWithType* aZoomWithType) 
	{ return aData.SetData(EZoomWithType, const_cast<TCoeZoomWithType*>(aZoomWithType)); }

// CustomGc
inline CWindowGc* DoGetCustomGc(const RCoeDynamicDataStorage& aData)
	{ return reinterpret_cast<CWindowGc*>(aData.Data(ECustomGc, NULL)); }

inline TInt DoSetCustomGc(const RCoeDynamicDataStorage& aData, CWindowGc* aGc) 
	{ return const_cast<RCoeDynamicDataStorage&>(aData).SetData(ECustomGc, aGc); }

// FontProvider
inline const CCoeFontProvider* DoGetFontProvider(const RCoeDynamicDataStorage& aData)
	{ return reinterpret_cast<const CCoeFontProvider*>(aData.Data(EFontProvider, NULL)); }

inline TInt DoSetFontProvider(const RCoeDynamicDataStorage& aData, const CCoeFontProvider* aFontProvider) 
	{ return const_cast<RCoeDynamicDataStorage&>(aData).SetData(EFontProvider, const_cast<CCoeFontProvider*>(aFontProvider)); }