Update change history to make Release 001 official.
// stringhash.h
// 
// 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
//
#ifndef FSHELL_STRINGHASH_H
#define FSHELL_STRINGHASH_H
#include <e32hashtab.h>
namespace LtkUtils
	{
	template <typename T> class TStringHashIter;
	template <typename T> struct SHashVal { HBufC* iKey; T iValue; };
	
	template <typename T>
	class RStringHash
		{
	public: // RHashMap-style
		inline RStringHash();
		inline T* Find(const TDesC& aKey);
		inline T FindPtr(const TDesC& aKey); // Only use if T is a pointer
		inline T& FindL(const TDesC& aKey);
		inline T const* Find(const TDesC& aKey) const;
		inline T const FindPtr(const TDesC& aKey) const; // Only use if T is a pointer
		inline T const& FindL(const TDesC& aKey) const;
		inline TInt Insert(const TDesC& aKey, const T& aValue);
		inline void InsertL(const TDesC& aKey, const T& aValue);
		inline TInt Remove(const TDesC& aKey);
		inline TInt Count() const;
		inline TInt Reserve(TInt aCount);
		inline void ReserveL(TInt aCount);
		inline void Close();
	protected:
		inline SHashVal<T>* DoFind(const TDesC& aKey);
		inline SHashVal<T> const* DoFind(const TDesC& aKey) const;
		friend class TStringHashIter<T>;
		RHashMap<TDesC*, SHashVal<T> > iHash;
		TAny* iSpare;
		TAny* iSpare2;
		};
	template <typename T>
	class TStringHashIter
		{
	public:
		inline TStringHashIter(const RStringHash<T>& aHash);
		inline const TDesC* CurrentKey() const;
		inline const TDesC* NextKey();
		inline T* CurrentValue();
		inline const T* NextValue();
		inline void RemoveCurrent();
	private:
		THashMapIter<TDesC*, SHashVal<T> > iIter;
		};
	inline TUint32 HashFn(TDesC* const& aKey);
	inline TBool IdFn(TDesC* const& aFirst, TDesC* const& aSecond);
	}
// Nothing to see here, move along move along
template <typename T>
LtkUtils::RStringHash<T>::RStringHash()
	: iHash(THashFunction32<TDesC*>(&HashFn), TIdentityRelation<TDesC*>(&IdFn)), iSpare(NULL), iSpare2(NULL)
	{
	}
inline TUint32 LtkUtils::HashFn(TDesC* const& aKey)
	{
	return DefaultHash::Des16(*aKey);
	}
inline TBool LtkUtils::IdFn(TDesC* const& aFirst, TDesC* const& aSecond)
	{
	return DefaultIdentity::Des16(*aFirst, *aSecond);
	}
template <typename T>
inline T* LtkUtils::RStringHash<T>::Find(const TDesC& aKey)
	{
	SHashVal<T>* res = DoFind(aKey);
	if (res) return &res->iValue;
	else return NULL;
	}
template <typename T>
inline T LtkUtils::RStringHash<T>::FindPtr(const TDesC& aKey)
	{
	T* res = Find(aKey);
	if (res) return *res;
	else return NULL;
	}
template <typename T>
inline T const* LtkUtils::RStringHash<T>::Find(const TDesC& aKey) const
	{
	SHashVal<T> const* res = DoFind(aKey);
	if (res) return &res->iValue;
	else return NULL;
	}
template <typename T>
inline T const LtkUtils::RStringHash<T>::FindPtr(const TDesC& aKey) const
	{
	T const* res = Find(aKey);
	if (res) return *res;
	else return NULL;
	}
template <typename T>
inline T& LtkUtils::RStringHash<T>::FindL(const TDesC& aKey)
	{
	T* res = Find(aKey);
	if (!res) User::Leave(KErrNotFound);
	return *res;
	}
template <typename T>
inline T const& LtkUtils::RStringHash<T>::FindL(const TDesC& aKey) const
	{
	T const* res = Find(aKey);
	if (!res) User::Leave(KErrNotFound);
	return *res;
	}
template <typename T>
TInt LtkUtils::RStringHash<T>::Insert(const TDesC& aKey, const T& aValue)
	{
	SHashVal<T>* valPtr = DoFind(aKey);
	if (valPtr)
		{
		// Just update the value, no need to realloc anything
		memcpy(&valPtr->iValue, &aValue, sizeof(T));
		return KErrNone;
		}
	HBufC* buf = aKey.Alloc();
	if (!buf) return KErrNoMemory;
	// Avoid declaring an SHashVal<T> as that requires T to have a default constructor
	TUint8 valBuf[sizeof(SHashVal<T>)];
	SHashVal<T>& val = *reinterpret_cast<SHashVal<T>*>(valBuf);
	val.iKey = buf;
	memcpy(&val.iValue, &aValue, sizeof(T));
	TInt err = iHash.Insert(buf, val);
	if (err)
		{
		delete buf;
		}
	return err;
	}
template <typename T>
inline TInt LtkUtils::RStringHash<T>::Remove(const TDesC& aKey)
	{
	SHashVal<T>* valPtr = DoFind(aKey);
	if (valPtr)
		{
		HBufC* key = valPtr->iKey;
		iHash.Remove(key);
		delete key;
		return KErrNone;
		}
	else
		{
		return KErrNotFound;
		}
	}
template <typename T>
inline void LtkUtils::RStringHash<T>::Close()
	{
	THashMapIter<TDesC*, SHashVal<T> > iter(iHash);
	while (iter.NextValue() != NULL)
		{
		SHashVal<T>* current = const_cast<SHashVal<T>*>(iter.CurrentValue()); // const_cast not needed except on 9.1
		HBufC* key = current->iKey;
		iter.RemoveCurrent();
		delete key;
		}
	iHash.Close();
	}
template <typename T>
inline void LtkUtils::RStringHash<T>::InsertL(const TDesC& aKey, const T& aValue)
	{
	User::LeaveIfError(Insert(aKey, aValue));
	}
template <typename T>
inline TInt LtkUtils::RStringHash<T>::Count() const
	{
	return iHash.Count();
	}
template <typename T>
inline TInt LtkUtils::RStringHash<T>::Reserve(TInt aCount)
	{
	return iHash.Reserve(aCount);
	}
template <typename T>
inline void LtkUtils::RStringHash<T>::ReserveL(TInt aCount)
	{
	User::LeaveIfError(Reserve(aCount));
	}
template <typename T>
inline LtkUtils::SHashVal<T>* LtkUtils::RStringHash<T>::DoFind(const TDesC& aKey)
	{
	return iHash.Find(const_cast<TDesC*>(&aKey));
	}
template <typename T>
inline LtkUtils::SHashVal<T> const* LtkUtils::RStringHash<T>::DoFind(const TDesC& aKey) const
	{
	return iHash.Find(const_cast<TDesC*>(&aKey));
	}
/// Iterator support
template <typename T>
inline const TDesC* LtkUtils::TStringHashIter<T>::CurrentKey() const
	{
	return *const_cast<TDesC**>(iIter.CurrentKey());
	}
template <typename T>
inline const TDesC* LtkUtils::TStringHashIter<T>::NextKey()
	{
	return *const_cast<TDesC**>(iIter.NextKey());
	}
template <typename T>
inline T* LtkUtils::TStringHashIter<T>::CurrentValue()
	{
	SHashVal<T>* val = const_cast<SHashVal<T>*>(iIter.CurrentValue()); // The const_cast isn't needed, except on 9.1
	if (val) return &val->iValue;
	return NULL;
	}
template <typename T>
inline const T* LtkUtils::TStringHashIter<T>::NextValue()
	{
	const SHashVal<T>* val = iIter.NextValue();
	if (val) return &val->iValue;
	return NULL;
	}
template <typename T>
inline void LtkUtils::TStringHashIter<T>::RemoveCurrent()
	{
	SHashVal<T>* val = iIter.CurrentValue();
	if (val)
		{
		HBufC* key = val->iKey;
		iIter.RemoveCurrent();
		delete key;
		}
	}
template <typename T>
inline LtkUtils::TStringHashIter<T>::TStringHashIter(const RStringHash<T>& aHash)
	: iIter(aHash.iHash)
	{}
#endif