|
|
/*==========================================================================
* * Copyright (C) 1999 Microsoft Corporation. All Rights Reserved. * * File: ClassHash.h * Content: Hash table that takes a class as a key. The key class MUST support * two member functions: * 'HashFunction' will perform a hash down to a specified number of bits. * 'CompareFunction' will perform a comparison of two items of that class. * * Note: This class requires an FPM to operate. * * THIS CLASS IS NOT THREAD SAFE!! * * History: * Date By Reason * ==== == ====== * 11/15/98 jwo Created it (map). * 04/19/99 jtk Rewrote without using STL (map) * 08/03/99 jtk Derived from ClassMap.h ***************************************************************************/
#ifndef __CLASS_HASH_H__
#define __CLASS_HASH_H__
#undef DPF_SUBCOMP
#define DPF_SUBCOMP DN_SUBCOMP_COMMON
//**********************************************************************
// Constant definitions
//**********************************************************************
//**********************************************************************
// Macro definitions
//**********************************************************************
#ifndef OFFSETOF
// Macro to compute the offset of an element inside of a larger structure (copied from MSDEV's STDLIB.H)
#define OFFSETOF(s,m) ( (INT_PTR) &(((s *)0)->m) )
#define __LOCAL_OFFSETOF_DEFINED__
#endif // OFFSETOF
//**********************************************************************
// Structure definitions
//**********************************************************************
//**********************************************************************
// Variable prototypes
//**********************************************************************
//**********************************************************************
// Function prototypes
//**********************************************************************
//**********************************************************************
// Class definitions
//**********************************************************************
//
// Template class for entry in map.
//
template<class T, class S> class CClassHashEntry { public: CClassHashEntry(){}; ~CClassHashEntry(){};
//
// internals, put the linkage at the end to make sure the FPM doesn't
// wail on it!
//
PVOID m_FPMPlaceHolder; S m_Key; T m_Item; CBilink m_Linkage;
//
// linkage functions
//
static CClassHashEntry *EntryFromBilink( CBilink *const pLinkage ) { DBG_CASSERT( sizeof( void* ) == sizeof( INT_PTR ) ); return reinterpret_cast<CClassHashEntry*>( &reinterpret_cast<BYTE*>( pLinkage )[ -OFFSETOF( CClassHashEntry, m_Linkage ) ] ); }
void AddToList( CBilink *const pLinkage ) { m_Linkage.InsertAfter( pLinkage ); }
void RemoveFromList( void ) { m_Linkage.RemoveFromList(); }
//
// pool management functions
//
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHashEntry::InitAlloc"
static BOOL InitAlloc( void *pItem ) { CClassHashEntry<T,S> *pThisObject;
DNASSERT( pItem != NULL ); pThisObject = static_cast<CClassHashEntry<T,S>*>( pItem );
pThisObject->m_Linkage.Initialize(); return TRUE; }
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHashEntry::Init"
static void Init( void *pItem ) { CClassHashEntry<T,S> *pThisObject;
DNASSERT( pItem != NULL ); pThisObject = static_cast<CClassHashEntry<T,S>*>( pItem ); DNASSERT( pThisObject->m_Linkage.IsEmpty() != FALSE ); }
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHashEntry::Release"
static void Release( void *pItem ) { CClassHashEntry<T,S> *pThisObject;
DNASSERT( pItem != NULL ); pThisObject = static_cast<CClassHashEntry<T,S>*>( pItem ); DNASSERT( pThisObject->m_Linkage.IsEmpty() != FALSE ); }
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHashEntry::Dealloc"
static void Dealloc( void *pItem ) { CClassHashEntry<T,S> *pThisObject;
DNASSERT( pItem != NULL ); pThisObject = static_cast<CClassHashEntry<T,S>*>( pItem ); DNASSERT( pThisObject->m_Linkage.IsEmpty() != FALSE ); }
protected:
private:
//
// make copy constructor and assignment operator private and unimplemented
// to prevent illegal copies from being made
//
CClassHashEntry( const CClassHashEntry & ); CClassHashEntry& operator=( const CClassHashEntry & ); };
//
// template class for the map
//
template<class T, class S> class CClassHash { public: CClassHash(); ~CClassHash();
BOOL Initialize( const INT_PTR iBitDepth, const INT_PTR iGrowBits ); void Deinitialize( void ); BOOL Insert( const S& Key, T Item ); void Remove( const S& Key ); BOOL RemoveLastEntry( T *const pItem ); BOOL Find( const S& Key, T *const pItem ); BOOL IsEmpty( void ) { return ( m_iEntriesInUse == 0 ); }
INT_PTR m_iHashBitDepth; // number of bits used for hash entry
INT_PTR m_iGrowBits; // number of bits to grow has by
CBilink *m_pHashEntries; // list of hash entries
INT_PTR m_iAllocatedEntries; // count of allocated entries in index/item list
INT_PTR m_iEntriesInUse; // count of entries in use
FPOOL m_EntryPool; // pool of entries
private: DEBUG_ONLY( BOOL m_fInitialized );
BOOL LocalFind( const S& Key, CBilink **const ppLink ); void Grow( void ); void InitializeHashEntries( const UINT_PTR uEntryCount ) const;
//
// make copy constructor and assignment operator private and unimplemented
// to prevent illegal copies from being made
//
CClassHash( const CClassHash & ); CClassHash& operator=( const CClassHash & ); };
//**********************************************************************
// Class function definitions
//**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::CClassHash - constructor
//
// Entry: Nothing
//
// Exit: Nothing
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::CClassHash"
template<class T, class S> CClassHash< T, S >::CClassHash(): m_iHashBitDepth( 0 ), m_iGrowBits( 0 ), m_pHashEntries( NULL ), m_iAllocatedEntries( 0 ), m_iEntriesInUse( 0 ) { //
// clear internals
//
DEBUG_ONLY( m_fInitialized = FALSE ); memset( &m_EntryPool, 0x00, sizeof( m_EntryPool ) ); } //**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::~CClassHash - destructor
//
// Entry: Nothing
//
// Exit: Nothing
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::~CClassHash"
template<class T, class S> CClassHash< T, S >::~CClassHash() { DNASSERT( m_iHashBitDepth == 0 ); DNASSERT( m_iGrowBits == 0 ); DNASSERT( m_pHashEntries == NULL ); DNASSERT( m_iAllocatedEntries == 0 ); DNASSERT( m_iEntriesInUse == 0 ); DEBUG_ONLY( DNASSERT( m_fInitialized == FALSE ) ); } //**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::Initialize - initialize hash table
//
// Entry: Pointer to key
// Pointer to 'key' associated with this item
// Pointer to item to add
//
// Exit: Boolean indicating success
// TRUE = success
// FALSE = failure
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::Initialize"
template<class T, class S> BOOL CClassHash< T, S >::Initialize( const INT_PTR iBitDepth, const INT_PTR iGrowBits ) { BOOL fReturn;
DNASSERT( iBitDepth != 0 );
//
// initialize
//
fReturn = TRUE;
DNASSERT( m_pHashEntries == NULL ); m_pHashEntries = static_cast<CBilink*>( DNMalloc( sizeof( *m_pHashEntries ) * ( 1 << iBitDepth ) ) ); if ( m_pHashEntries == NULL ) { fReturn = FALSE; DPFX(DPFPREP, 0, "Unable to allocate memory for hash table!" ); goto Exit; } m_iAllocatedEntries = 1 << iBitDepth; InitializeHashEntries( m_iAllocatedEntries );
if ( FPM_Initialize( &m_EntryPool, // pointer to pool
sizeof( CClassHashEntry<T,S> ), // size of pool entry
CClassHashEntry<T,S>::InitAlloc, // function for allocating item
CClassHashEntry<T,S>::Init, // function for getting item from pool
CClassHashEntry<T,S>::Release, // function for releasing item
CClassHashEntry<T,S>::Dealloc // function for deallocating item
) == FALSE ) { DPFX(DPFPREP, 0, "Failed to initialize FPM!" ); fReturn = FALSE; }
m_iHashBitDepth = iBitDepth; m_iGrowBits = iGrowBits;
DEBUG_ONLY( m_fInitialized = TRUE );
Exit: return fReturn; } //**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::Deinitialize - deinitialize hash table
//
// Entry: Nothing
//
// Exit: Nothing
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::Deinitialize"
template<class T, class S> void CClassHash< T, S >::Deinitialize( void ) { DEBUG_ONLY( DNASSERT( m_fInitialized != FALSE ) ); DNASSERT( m_iEntriesInUse == 0 ); DNASSERT( m_pHashEntries != NULL ); DNFree( m_pHashEntries ); m_pHashEntries = NULL; FPM_Deinitialize( &m_EntryPool ); m_iHashBitDepth = 0; m_iGrowBits = 0; m_iAllocatedEntries = 0; DEBUG_ONLY( m_fInitialized = FALSE ); } //**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::Insert - add item to map
//
// Entry: Pointer to 'key' associated with this item
// Pointer to item to add
//
// Exit: Boolean indicating success:
// TRUE = success
// FALSE = failure
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::Insert"
template<class T, class S> BOOL CClassHash< T, S >::Insert( const S& Key, T Item ) { BOOL fReturn; BOOL fFound; CBilink *pLink; CClassHashEntry< T, S > *pNewEntry;
DEBUG_ONLY( DNASSERT( m_fInitialized != FALSE ) );
//
// initialize
//
fReturn = TRUE; pNewEntry = NULL;
//
// grow the map if applicable
//
if ( ( m_iEntriesInUse >= ( m_iAllocatedEntries / 2 ) ) && ( m_iGrowBits != 0 ) ) { Grow(); }
//
// get a new table entry before trying the lookup
//
pNewEntry = static_cast<CClassHashEntry<T,S>*>( m_EntryPool.Get( &m_EntryPool ) ); if ( pNewEntry == NULL ) { fReturn = FALSE; DPFX(DPFPREP, 0, "Problem allocating new hash table entry on Insert!" ); goto Exit; }
//
// scan for this item in the list, since we're only supposed to have
// unique items in the list, ASSERT if a duplicate is found
//
fFound = LocalFind( Key, &pLink ); DNASSERT( pLink != NULL ); DNASSERT( fFound == FALSE );
//
// officially add entry to the hash table
//
m_iEntriesInUse++; pNewEntry->m_Key = Key; pNewEntry->m_Item = Item; DNASSERT( pLink != NULL ); pNewEntry->AddToList( pLink );
DNASSERT( fReturn == TRUE );
Exit: return fReturn; } //**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::Remove - remove item from map
//
// Entry: Reference to 'key' used to look up this item
//
// Exit: Nothing
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::Remove"
template<class T, class S> void CClassHash< T, S >::Remove( const S& Key ) { CBilink *pLink;
DEBUG_ONLY( DNASSERT( m_fInitialized != FALSE ) ); if ( LocalFind( Key, &pLink ) != FALSE ) { CClassHashEntry< T, S > *pEntry;
DNASSERT( pLink != NULL ); pEntry = CClassHashEntry< T, S >::EntryFromBilink( pLink ); pEntry->RemoveFromList(); m_EntryPool.Release( &m_EntryPool, pEntry );
DNASSERT( m_iEntriesInUse != 0 ); m_iEntriesInUse--; } } //**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::RemoveLastEntry - remove last item from map
//
// Entry: Pointer to pointer to item data
//
// Exit: Boolean indicating success
// TRUE = item was removed
// FALSE = item was not removed (map empty)
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::RemoveLastEntry"
template<class T, class S> BOOL CClassHash< T, S >::RemoveLastEntry( T *const pItem ) { BOOL fReturn;
DNASSERT( pItem != NULL );
//
// initialize
//
DEBUG_ONLY( DNASSERT( m_fInitialized != FALSE ) ); fReturn = FALSE;
if ( m_iEntriesInUse != 0 ) { INT_PTR iIndex;
DNASSERT( m_pHashEntries != NULL ); iIndex = m_iAllocatedEntries; while ( iIndex > 0 ) { iIndex--;
if ( m_pHashEntries[ iIndex ].IsEmpty() == FALSE ) { CClassHashEntry<T,S> *pEntry;
pEntry = pEntry->EntryFromBilink( m_pHashEntries[ iIndex ].GetNext() ); pEntry->RemoveFromList(); *pItem = pEntry->m_Item; m_EntryPool.Release( &m_EntryPool, pEntry ); m_iEntriesInUse--; fReturn = TRUE;
goto Exit; } } }
Exit: return fReturn; } //**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::Find - find item in map
//
// Entry: Reference of 'key' used to look up this item
// Pointer to pointer to be filled in with data
//
// Exit: Boolean indicating success
// TRUE = item found
// FALSE = item not found
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::Find"
template<class T, class S> BOOL CClassHash< T, S >::Find( const S& Key, T *const pItem ) { BOOL fReturn; CBilink *pLinkage;
DEBUG_ONLY( DNASSERT( m_fInitialized != FALSE ) );
//
// initialize
//
fReturn = FALSE; pLinkage = NULL;
if ( LocalFind( Key, &pLinkage ) != FALSE ) { CClassHashEntry<T,S> *pEntry;
pEntry = CClassHashEntry<T,S>::EntryFromBilink( pLinkage ); *pItem = pEntry->m_Item; fReturn = TRUE; }
return fReturn; } //**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::LocalFind - find an entry in a hash table, or find out where to insert.
//
// Entry: Refernce of 'key' to look for
// Pointer to pointer to linkage of find or insert
//
// Exit: Boolean indicating whether the item was found
// TRUE = found
// FALSE = not found
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::LocalFind"
template<class T, class S> BOOL CClassHash< T, S >::LocalFind( const S& Key, CBilink **const ppLinkage ) { BOOL fFound; INT_PTR HashResult; CBilink *pTemp;
DEBUG_ONLY( DNASSERT( m_fInitialized != FALSE ) );
HashResult = Key->HashFunction( m_iHashBitDepth ); DNASSERT( ( HashResult < ( 1 << m_iHashBitDepth ) ) && ( HashResult >= 0 ) );
fFound = FALSE; pTemp = &m_pHashEntries[ HashResult ]; while ( pTemp->GetNext() != &m_pHashEntries[ HashResult ] ) { const CClassHashEntry< T, S > *pEntry;
pEntry = CClassHashEntry< T, S >::EntryFromBilink( pTemp->GetNext() ); if ( Key->CompareFunction( pEntry->m_Key ) == 0 ) { fFound = TRUE; *ppLinkage = pTemp->GetNext(); goto Exit; } else { pTemp = pTemp->GetNext(); } }
//
// entry was not found, return pointer to linkage to insert after if a new
// entry is being added to the table
//
*ppLinkage = pTemp;
Exit: return fFound; } //**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::Grow - grow hash table to next larger size
//
// Entry: Nothing
//
// Exit: Nothing
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::Grow"
template<class T, class S> void CClassHash< T, S >::Grow( void ) { CBilink *pTemp; INT_PTR iNewEntryBitCount;
DEBUG_ONLY( DNASSERT( m_fInitialized != FALSE ) ); DNASSERT( m_iGrowBits != 0 );
//
// We're more than 50% full, find a new has table size that will accomodate
// all of the current entries, and keep a pointer to the old data in case
// the memory allocation fails.
//
pTemp = m_pHashEntries; iNewEntryBitCount = m_iHashBitDepth;
do { iNewEntryBitCount += m_iGrowBits; } while ( m_iEntriesInUse >= ( ( 1 << iNewEntryBitCount ) / 2 ) );
//
// assert that we don't pull up half of the machine's address space!
//
DNASSERT( iNewEntryBitCount <= ( sizeof( UINT_PTR ) * 8 / 2 ) );
m_pHashEntries = static_cast<CBilink*>( DNMalloc( sizeof( *pTemp ) * ( 1 << iNewEntryBitCount ) ) ); if ( m_pHashEntries == NULL ) { //
// Allocation failed, restore the old data pointer and insert the item
// into the hash table. This will probably result in adding to a bucket.
//
m_pHashEntries = pTemp; DPFX(DPFPREP, 0, "Warning: Failed to grow hash table when 50% full!" ); } else { INT_PTR iOldHashSize; DEBUG_ONLY( INT_PTR iOldEntryCount );
//
// we have more memory, reorient the hash table and re-add all of
// the old items
//
InitializeHashEntries( 1 << iNewEntryBitCount ); DEBUG_ONLY( iOldEntryCount = m_iEntriesInUse );
iOldHashSize = 1 << m_iHashBitDepth; m_iHashBitDepth = iNewEntryBitCount;
m_iAllocatedEntries = 1 << iNewEntryBitCount; m_iEntriesInUse = 0;
DNASSERT( iOldHashSize > 0 ); while ( iOldHashSize > 0 ) { iOldHashSize--; while ( pTemp[ iOldHashSize ].GetNext() != &pTemp[ iOldHashSize ] ) { BOOL fTempReturn; S Key; T Item; CClassHashEntry<T,S> *pTempEntry;
pTempEntry = pTempEntry->EntryFromBilink( pTemp[ iOldHashSize ].GetNext() ); pTempEntry->RemoveFromList(); Key = pTempEntry->m_Key; Item = pTempEntry->m_Item; m_EntryPool.Release( &m_EntryPool, pTempEntry );
//
// Since we're returning the current hash table entry to the pool
// it will be immediately reused in the new table. We should never
// have a problem adding to the new list.
//
fTempReturn = Insert( Key, Item ); DNASSERT( fTempReturn != FALSE ); DEBUG_ONLY( iOldEntryCount-- ); } }
DEBUG_ONLY( DNASSERT( iOldEntryCount == 0 ) ); DNFree( pTemp ); pTemp = NULL; } } //**********************************************************************
//**********************************************************************
// ------------------------------
// CClassHash::InitializeHashEntries - initialize all of the entries in the hash table
//
// Entry: Count of entries to initialize.
//
// Exit: Nothing
// ------------------------------
#undef DPF_MODNAME
#define DPF_MODNAME "CClassHash::InitializeHashEntries"
template<class T, class S> void CClassHash< T, S >::InitializeHashEntries( const UINT_PTR uEntryCount ) const { UINT_PTR uLocalEntryCount;
DNASSERT( m_pHashEntries != NULL ); uLocalEntryCount = uEntryCount; while ( uLocalEntryCount != 0 ) { uLocalEntryCount--;
m_pHashEntries[ uLocalEntryCount ].Initialize(); } } //**********************************************************************
#ifdef __LOCAL_OFFSETOF_DEFINED__
#undef __LOCAL_OFFSETOF_DEFINED__
#undef OFFSETOF
#endif // __LOCAL_OFFSETOF_DEFINED__
#undef DPF_SUBCOMP
#undef DPF_MODNAME
#endif // __CLASS_HASH_H__
|