//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============// // // Purpose: // //=============================================================================// #ifndef SAVERESTORE_UTLMAP_H #define SAVERESTORE_UTLMAP_H #include "utlmap.h" #include "saverestore_utlrbtree.h" #if defined( _WIN32 ) #pragma once #endif template class CUtlMapDataOps : public CDefSaveRestoreOps { public: CUtlMapDataOps() { // These COMPILE_TIME_ASSERT checks need to be in individual scopes to avoid build breaks // on MacOS and Linux due to a gcc bug. { UTLCLASS_SAVERESTORE_VALIDATE_TYPE( KEY_TYPE ); } { UTLCLASS_SAVERESTORE_VALIDATE_TYPE( FIELD_TYPE ); } } virtual void Save( const SaveRestoreFieldInfo_t &fieldInfo, ISave *pSave ) { datamap_t *pKeyDatamap = CTypedescDeducer::Deduce( (UTLMAP *)NULL ); datamap_t *pFieldDatamap = CTypedescDeducer::Deduce( (UTLMAP *)NULL ); typedescription_t dataDesc[] = { { (fieldtype_t)KEY_TYPE, "K", 0, 1, FTYPEDESC_SAVE, NULL, NULL, NULL, pKeyDatamap, sizeof(KEY_TYPE), }, { (fieldtype_t)FIELD_TYPE, "T", offsetof(typename UTLMAP::Node_t, elem), 1, FTYPEDESC_SAVE, NULL, NULL, NULL, pFieldDatamap, sizeof(FIELD_TYPE), } }; datamap_t dataMap = { dataDesc, 2, "um", NULL, 0, NULL, #ifdef _DEBUG true #endif }; typename UTLMAP::CTree *pUtlRBTree = ((UTLMAP *)fieldInfo.pField)->AccessTree(); pSave->StartBlock(); int nElems = pUtlRBTree->Count(); pSave->WriteInt( &nElems, 1 ); typename UTLMAP::CTree::IndexType_t i = pUtlRBTree->FirstInorder(); while ( i != pUtlRBTree->InvalidIndex() ) { typename UTLMAP::CTree::ElemType_t &elem = pUtlRBTree->Element( i ); pSave->WriteAll( &elem, &dataMap ); i = pUtlRBTree->NextInorder( i ); } pSave->EndBlock(); } virtual void Restore( const SaveRestoreFieldInfo_t &fieldInfo, IRestore *pRestore ) { datamap_t *pKeyDatamap = CTypedescDeducer::Deduce( (UTLMAP *)NULL ); datamap_t *pFieldDatamap = CTypedescDeducer::Deduce( (UTLMAP *)NULL ); typedescription_t dataDesc[] = { { (fieldtype_t)KEY_TYPE, "K", 0, 1, FTYPEDESC_SAVE, NULL, NULL, NULL, pKeyDatamap, sizeof(KEY_TYPE), }, { (fieldtype_t)FIELD_TYPE, "T", offsetof(typename UTLMAP::Node_t, elem), 1, FTYPEDESC_SAVE, NULL, NULL, NULL, pFieldDatamap, sizeof(FIELD_TYPE), } }; datamap_t dataMap = { dataDesc, 2, "um", NULL, 0, NULL, #ifdef _DEBUG true #endif }; UTLMAP *pUtlMap = ((UTLMAP *)fieldInfo.pField); pRestore->StartBlock(); int nElems = pRestore->ReadInt(); typename UTLMAP::CTree::ElemType_t temp; while ( nElems-- ) { pRestore->ReadAll( &temp, &dataMap ); pUtlMap->Insert( temp.key, temp.elem ); } pRestore->EndBlock(); } virtual void MakeEmpty( const SaveRestoreFieldInfo_t &fieldInfo ) { UTLMAP *pUtlMap = (UTLMAP *)fieldInfo.pField; pUtlMap->RemoveAll(); } virtual bool IsEmpty( const SaveRestoreFieldInfo_t &fieldInfo ) { UTLMAP *pUtlMap = (UTLMAP *)fieldInfo.pField; return ( pUtlMap->Count() == 0 ); } }; //------------------------------------- template class CUtlMapDataopsInstantiator { public: template static ISaveRestoreOps *GetDataOps(UTLMAP *) { static CUtlMapDataOps ops; return &ops; } }; //------------------------------------- #define SaveUtlMap( pSave, pUtlMap, fieldtype) \ CUtlMapDataopsInstantiator::GetDataOps( pUtlMap )->Save( pUtlMap, pSave ); #define RestoreUtlMap( pRestore, pUtlMap, fieldtype) \ CUtlMapDataopsInstantiator::GetDataOps( pUtlMap )->Restore( pUtlMap, pRestore ); //------------------------------------- #define DEFINE_UTLMAP(name,keyType,fieldtype) \ { FIELD_CUSTOM, #name, offsetof(classNameTypedef,name), 1, FTYPEDESC_SAVE, NULL, CUtlMapDataopsInstantiator::GetDataOps(&(((classNameTypedef *)0)->name)), NULL } #endif // SAVERESTORE_UTLMAP_H