/*==LICENSE==*

CyanWorlds.com Engine - MMOG client, server and tools
Copyright (C) 2011  Cyan Worlds, Inc.

This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program.  If not, see <http://www.gnu.org/licenses/>.

Additional permissions under GNU GPL version 3 section 7

If you modify this Program, or any covered work, by linking or
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
(or a modified version of those libraries),
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
licensors of this Program grant you additional
permission to convey the resulting work. Corresponding Source for a
non-source form of such a combination shall include the source code for
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
work.

You can contact Cyan Worlds, Inc. by email legal@cyan.com
 or by snail mail at:
      Cyan Worlds, Inc.
      14617 N Newport Hwy
      Mead, WA   99021

*==LICENSE==*/
//////////////////////////////////////////////////////////////////////////////
//
//  plResManagerHelper - The wonderful helper class that can receive messages
//                       for the resManager.
//
//// History /////////////////////////////////////////////////////////////////
//
//  6.7.2002 mcn    - Created
//
//////////////////////////////////////////////////////////////////////////////

#include "HeadSpin.h"
#include "plResManagerHelper.h"
#include "plResManager.h"
#include "plRegistryNode.h"
#include "plRegistryHelpers.h"
//#include "plRegistry.h"
#include "plResMgrSettings.h"

#include "pnKeyedObject/plFixedKey.h"
#include "plMessage/plResMgrHelperMsg.h"
#include "plStatusLog/plStatusLog.h"
#include "hsTimer.h"

#ifdef MCN_RESMGR_DEBUGGING

static const int    kLogSize        = 40;
static const float  kUpdateDelay    = 0.5f;

#include "plInputCore/plInputInterface.h"
#include "plInputCore/plInputDevice.h"
#include "plInputCore/plInputInterfaceMgr.h"
#include "pnInputCore/plInputMap.h"
#include "plMessage/plInputEventMsg.h"
#include "plMessage/plInputIfaceMgrMsg.h"
#include "pnKeyedObject/plKeyImp.h"

#endif

/// Logging #define for easier use
#define kResMgrLog( level ) if( plResMgrSettings::Get().GetLoggingLevel() >= level ) plStatusLog::AddLineS( "resources.log", 



//// Constructor/Destructor //////////////////////////////////////////////////

plResManagerHelper  *plResManagerHelper::fInstance = nil;

plResManagerHelper::plResManagerHelper( plResManager *resMgr )
{
    fResManager = resMgr;
    fInstance = this;

    fInShutdown = false;
#ifdef MCN_RESMGR_DEBUGGING
    fDebugScreen = nil;
    fCurrAge = -1;
    fCurrAgeExpanded = false;
    fRefreshing = false;
    fDebugDisplayType = 0;
#endif
}
plResManagerHelper::~plResManagerHelper()
{
    fInstance = nil;
}

//// Shutdown ////////////////////////////////////////////////////////////////

void    plResManagerHelper::Shutdown( void )
{
    EnableDebugScreen( false );
    UnRegisterAs( kResManagerHelper_KEY );
}

//// Init ////////////////////////////////////////////////////////////////////

void    plResManagerHelper::Init( void )
{
    RegisterAs( kResManagerHelper_KEY );
}

//// MsgReceive //////////////////////////////////////////////////////////////

bool    plResManagerHelper::MsgReceive( plMessage *msg )
{
    plResMgrHelperMsg *refferMsg = plResMgrHelperMsg::ConvertNoRef( msg );
    if( refferMsg != nil )
    {
        if( refferMsg->GetCommand() == plResMgrHelperMsg::kKeyRefList )
        {
            // Message to let go of these keys. So unref the key list, destroy it and we're done!
            kResMgrLog( 2 ) 0xff80ff80, "Dropping page keys after timed delay" );
            hsStatusMessage( "*** Dropping page keys after timed delay ***" );

            delete refferMsg->fKeyList;
            refferMsg->fKeyList = nil;
        }
        else if( refferMsg->GetCommand() == plResMgrHelperMsg::kUpdateDebugScreen )
        {
            IUpdateDebugScreen();
        }
        else if( refferMsg->GetCommand() == plResMgrHelperMsg::kEnableDebugScreen )
            EnableDebugScreen( true );
        else if( refferMsg->GetCommand() == plResMgrHelperMsg::kDisableDebugScreen )
            EnableDebugScreen( false );

        return true;
    }

    return hsKeyedObject::MsgReceive( msg );
}

//// Read/Write //////////////////////////////////////////////////////////////
    
void    plResManagerHelper::Read( hsStream *s, hsResMgr *mgr )
{
    hsAssert( false, "You should never read me in!" );
}

void    plResManagerHelper::Write( hsStream *s, hsResMgr *mgr )
{
    hsAssert( false, "You should never write me out!" );
}

//// LoadAndHoldPageKeys /////////////////////////////////////////////////////
//  Loads and refs the keys for the given page, then sends the ref list as
//  a list to ourself, time delayed 1 second, so that we can unref them one
//  second later.

void plResManagerHelper::LoadAndHoldPageKeys( plRegistryPageNode *page )
{
    hsAssert( GetKey() != nil, "Can't load and hold keys when we don't have a key for the helper" );

    // Create our msg
    plResMgrHelperMsg   *refferMsg = new plResMgrHelperMsg( plResMgrHelperMsg::kKeyRefList );
    refferMsg->fKeyList = new plResPageKeyRefList;

    fResManager->LoadPageKeys(page);
    page->IterateKeys( refferMsg->fKeyList );

    // Load and ref the keys
#ifdef HS_DEBUGGING
    char msg[ 256 ];
    sprintf( msg, "*** Temporarily loading keys for room %s>%s based on FindKey() query, will drop in 1 sec ***", page->GetPageInfo().GetAge().c_str(), page->GetPageInfo().GetPage().c_str());
    hsStatusMessage( msg );
#endif
    kResMgrLog( 2 ) 0xff80ff80, "Temporarily loading keys for room %s>%s, will drop in 1 sec", page->GetPageInfo().GetAge().c_str(), page->GetPageInfo().GetPage().c_str());
        
    // Deliver the message to ourselves!
    refferMsg->SetTimeStamp( hsTimer::GetSysSeconds() + 1.f );
    refferMsg->Send( GetKey() );
}

#ifdef MCN_RESMGR_DEBUGGING

//////////////////////////////////////////////////////////////////////////////
//// plResMgrDebugInterface Definition ///////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////

class plResMgrDebugInterface : public plInputInterface
{
    protected:
        plResManagerHelper  * const fParent;

        virtual ControlEventCode    *IGetOwnedCodeList( void ) const
        {
            static ControlEventCode codes[] = { END_CONTROLS };
            return codes;
        }

    public:

        plResMgrDebugInterface( plResManagerHelper * const mgr ) : fParent( mgr ) { SetEnabled( true ); }

        virtual uint32_t  GetPriorityLevel( void ) const { return kGUISystemPriority + 10; }
        virtual bool    InterpretInputEvent( plInputEventMsg *pMsg )
        {
            plKeyEventMsg *pKeyMsg = plKeyEventMsg::ConvertNoRef( pMsg );
            if( pKeyMsg != nil && pKeyMsg->GetKeyDown() )
            {
                if( pKeyMsg->GetKeyCode() == KEY_UP && fParent->fCurrAge >= 0 )
                {
                    fParent->fCurrAge--;
                    fParent->IUpdateDebugScreen( true );
                    return true;
                }
                else if( pKeyMsg->GetKeyCode() == KEY_DOWN )
                {
                    fParent->fCurrAge++;
                    fParent->IUpdateDebugScreen( true );
                    return true;
                }
                else if( pKeyMsg->GetKeyCode() == KEY_ENTER )
                {
                    fParent->fCurrAgeExpanded = !fParent->fCurrAgeExpanded;
                    fParent->IUpdateDebugScreen( true );
                    return true;
                }
                else if( pKeyMsg->GetKeyCode() == KEY_ESCAPE )
                {
                    plResMgrHelperMsg *msg = new plResMgrHelperMsg( plResMgrHelperMsg::kDisableDebugScreen );
                    msg->Send( fParent->GetKey() );
                    return true;
                }
                else if( pKeyMsg->GetKeyCode() == KEY_RIGHT )
                {
                    if( !fParent->fCurrAgeExpanded )
                        fParent->fCurrAgeExpanded = true;
                    else
                    {
                        fParent->fDebugDisplayType++;
                        if( fParent->fDebugDisplayType == plResManagerHelper::kMaxDisplayType )
                            fParent->fDebugDisplayType = 0;
                    }
                    fParent->IUpdateDebugScreen( true );
                    return true;
                }
                else if( pKeyMsg->GetKeyCode() == KEY_LEFT )
                {
                    fParent->fCurrAgeExpanded = false;
                    fParent->IUpdateDebugScreen( true );
                    return true;
                }
            }

            return false;
        }

        virtual uint32_t  GetCurrentCursorID( void ) const { return 0; }
        virtual bool    HasInterestingCursorID( void ) const { return false; }
};

#endif

//// EnableDebugScreen ///////////////////////////////////////////////////////

void    plResManagerHelper::EnableDebugScreen( bool enable )
{
#ifdef MCN_RESMGR_DEBUGGING
    if( enable )
    {
        if( fDebugScreen == nil )
        {
            fDebugScreen = plStatusLogMgr::GetInstance().CreateStatusLog( kLogSize, "ResManager Status", plStatusLog::kFilledBackground | plStatusLog::kDontWriteFile );
            fRefreshing = true;

            plResMgrHelperMsg *msg = new plResMgrHelperMsg( plResMgrHelperMsg::kUpdateDebugScreen );
//          msg->SetTimeStamp( hsTimer::GetSysSeconds() + kUpdateDelay );
            msg->Send( GetKey() );

            fDebugInput = new plResMgrDebugInterface( this );
            plInputIfaceMgrMsg *imsg = new plInputIfaceMgrMsg( plInputIfaceMgrMsg::kAddInterface );
            imsg->SetIFace( fDebugInput );
            imsg->Send();
        }
    }
    else
    {
        fRefreshing = false;
        if( fDebugScreen != nil )
        {
            delete fDebugScreen;
            fDebugScreen = nil;

            plInputIfaceMgrMsg *imsg = new plInputIfaceMgrMsg( plInputIfaceMgrMsg::kRemoveInterface );
            imsg->SetIFace( fDebugInput );
            imsg->Send();

            hsRefCnt_SafeUnRef( fDebugInput );
            fDebugInput = nil;
        }
    }
#endif
}

//// IUpdateDebugScreen  /////////////////////////////////////////////////////

#ifdef MCN_RESMGR_DEBUGGING
class plDebugPrintIterator : public plRegistryPageIterator, plRegistryKeyIterator
{   
    public:
        plStatusLog *fLog;
        uint8_t       fStep, fLines;
        uint32_t      &fLoadedCount, &fHoldingCount, fPageCount, fAgeIndex;
        char        fCurrAge[ 128 ];
        uint32_t      fLoadedKeys, fTotalKeys, fTotalSize, fLoadedSize;

        plResManagerHelper  *fParent;

        plDebugPrintIterator( plResManagerHelper *parent, plStatusLog *log, uint32_t &loadedCount, uint32_t &holdingCount ) 
                    : fParent( parent ), fLog( log ), fStep( 0 ), fLines( 0 ), fLoadedCount( loadedCount ), fHoldingCount( holdingCount )
        {
            fLoadedCount = fHoldingCount = 0;
            fCurrAge[ 0 ] = 0;
            fPageCount = 0;
            fAgeIndex = 0;
        }

        virtual bool    EatPage( plRegistryPageNode *page )
        {
            if( fStep == 0 )
            {
                fLog->AddLineF( 0xff80ff80, "Loaded Pages" );
                fStep = 1;
                fLines++;
            }
            else if( fStep == 1 && page != nil && !page->IsLoaded() )
            {
                fStep = 2;
                fLog->AddLineF( 0xff80ff80, "Holding Pages" );
                fLines++;
            }
    
            if( page != nil && page->IsLoaded() )
                fLoadedCount++;
            else if( page != nil )
                fHoldingCount++;

            // Changed ages?
            if( page == nil || page->GetPageInfo().GetAge() != fCurrAge )
            {
                // Print some info for the last age we were on
                if( fCurrAge[ 0 ] != 0 )
                {
                    if( fParent->fCurrAge != fAgeIndex || !fParent->fCurrAgeExpanded )
                    {
                        if( fLines < kLogSize - 4 )
                        {
                            uint32_t color = plStatusLog::kWhite;
                            if( fParent->fCurrAge == fAgeIndex )
                                color = plStatusLog::kYellow;

                            fLog->AddLineF( color, " %s (%d pages)", fCurrAge, fPageCount );
                            fLines++;
                        }
                        else if( fLines == kLogSize - 4 )
                        {
                            fLog->AddLineF( plStatusLog::kWhite, " ..." );
                            fLines++;
                        }
                    }
                    fAgeIndex++;
                }
                fPageCount = 0;
                if( page != nil )
                    strncpy( fCurrAge, page->GetPageInfo().GetAge().c_str(), sizeof( fCurrAge ) - 1 );
                else
                    fCurrAge[ 0 ] = 0;

                if( fParent->fCurrAge == fAgeIndex && fParent->fCurrAgeExpanded )
                {
                    // Print header now, since we won't be printing a footer
                    if( fLines < kLogSize - 4 )
                    {
                        fLog->AddLineF( plStatusLog::kYellow, " %s>", fCurrAge );   
                        fLines++;
                    }
                    else if( fLines == kLogSize - 4 )
                    {
                        fLog->AddLineF( plStatusLog::kWhite, " ..." );
                        fLines++;
                    }
                }
            }

            fPageCount++;

            if( fParent->fCurrAge == fAgeIndex && fParent->fCurrAgeExpanded && page != nil )
            {
                // Count keys for this page
                fTotalKeys = fLoadedKeys = fTotalSize = fLoadedSize = 0;
                page->IterateKeys( this );

                // Print page for this expanded age view
                if( fLines < kLogSize - 4 )
                {
                    if( fParent->fDebugDisplayType == plResManagerHelper::kSizes )
                        fLog->AddLineF( plStatusLog::kWhite, "  %s (%d keys @ %4.1fk, %d loaded @ %4.1fk)", page->GetPageInfo().GetPage().c_str(), fTotalKeys, fTotalSize / 1024.f, fLoadedKeys, fLoadedSize / 1024.f );
                    else if( fParent->fDebugDisplayType == plResManagerHelper::kPercents )
                        fLog->AddLineF( plStatusLog::kWhite, "  %s (%d%% loaded of %d keys @ %4.1fk)", page->GetPageInfo().GetPage().c_str(), fLoadedSize * 100 / ( fTotalSize > 0 ? fTotalSize : -1 ), fTotalKeys, fTotalSize / 1024.f );
                    else //if( fParent->fDebugDisplayType == plResManagerHelper::kBars )
                    {
                        const int startPos = 20, length = 32;

                        char line[ 128 ];
                        memset( line, ' ', sizeof( line ) - 1 );
                        line[ 127 ] = 0;
                        if(page->GetPageInfo().GetPage().GetSize() < startPos - 2 )
                            memcpy( line + 2, page->GetPageInfo().GetPage().c_str(), page->GetPageInfo().GetPage().GetSize() );
                        else
                            memcpy( line + 2, page->GetPageInfo().GetPage().c_str(), startPos - 2 );

                        line[ startPos ] = '|';
                        if( fTotalSize == 0 )
                        {
                            line[ startPos + 1 ] = '|';
                            line[ startPos + 2 ] = 0;
                        }
                        else
                        {
                            char temp[ 12 ];
                            sprintf( temp, "%d%%", fLoadedSize * 100 / fTotalSize );

                            line[ startPos + length + 1 ] = '|';
                            int i, sum = 0;
                            for( i = startPos + 1; i < startPos + length + 1 && sum < fLoadedSize; i++ )
                            {
                                line[ i ] = '=';
                                sum += fTotalSize / length;
                            }
                            line[ startPos + length + 2 ] = 0;

                            memcpy( line + startPos + 1, temp, strlen( temp ) );
                        }

                        fLog->AddLine( line, plStatusLog::kWhite );
                    }
                    fLines++;
                }
                else if( fLines == kLogSize - 4 )
                {
                    fLog->AddLineF( plStatusLog::kWhite, " ..." );
                    fLines++;
                }   
            }

            return true;
        }

        virtual bool    EatKey( const plKey& key )
        {
            if( key->ObjectIsLoaded() )
            {
                fLoadedKeys++;
                fLoadedSize += ((plKeyImp *)key)->GetDataLen();
            }
            fTotalKeys++;
            fTotalSize += ((plKeyImp *)key)->GetDataLen();
            return true;
        }
};
#endif

void    plResManagerHelper::IUpdateDebugScreen( bool force )
{
#ifdef MCN_RESMGR_DEBUGGING

    if( !fRefreshing )
        return;

    uint32_t      loadedCnt, holdingCnt;

    fDebugScreen->Clear();

    plDebugPrintIterator    iter( this, fDebugScreen, loadedCnt, holdingCnt );
    fResManager->IterateAllPages( &iter );
    iter.EatPage( nil );        // Force a final update

    fDebugScreen->AddLineF( plStatusLog::kGreen, "%d pages loaded, %d holding", loadedCnt, holdingCnt );

    if( fCurrAge >= iter.fAgeIndex )
        fCurrAge = -1;

    // Repump our update
    if( !force )
    {
        plResMgrHelperMsg *msg = new plResMgrHelperMsg( plResMgrHelperMsg::kUpdateDebugScreen );
        msg->SetTimeStamp( hsTimer::GetSysSeconds() + kUpdateDelay );
        msg->Send( GetKey() );
    }

#endif
}

#if 0
// FIXME
    bool VerifyKeyUnloaded(const char* logFile, const plKey& key);
    // Verifies that a key which shouldn't be loaded isn't, and if it is tries to figure out why.
    void VerifyAgeUnloaded(const char* logFile, const char* age);

    // Helper for VerifyKeyUnloaded
    bool    IVerifyKeyUnloadedRecur(const char* logFile, const plKey& baseKey, const plKey& upKey, const char* baseAge);
    bool ILookForCyclesRecur(const char* logFile, const plKey& key, hsTArray<plKey>& tree, int& cycleStart);

bool plResManager::ILookForCyclesRecur(const char* logFile, const plKey& key, hsTArray<plKey>& tree, int& cycleStart)
{
    int idx = tree.Find(key);
    tree.Append(key);
    if (tree.kMissingIndex != idx)
    {
        cycleStart = idx;
        // Found a cycle.
        return true;
    }

    // Now recurse up the active reference tree.
    for (int i = 0; i < key->GetNumNotifyCreated(); i++)
    {
        if (key->GetActiveBits().IsBitSet(i))
        {
            for (int j = 0; j < key->GetNotifyCreated(i)->GetNumReceivers(); j++)
            {
                plKey reffer = key->GetNotifyCreated(i)->GetReceiver(j);
                
                if (ILookForCyclesRecur(logFile, reffer, tree, cycleStart))
                    return true;
            }
        }
    }
    tree.Pop();
    return false;
}

bool plResManager::IVerifyKeyUnloadedRecur(const char* logFile, const plKey& baseKey, const plKey& upKey, const char* baseAge)
{
    const plPageInfo& pageInfo = FindPage(upKey->GetUoid().GetLocation())->GetPageInfo();
    const char* upAge = pageInfo.GetAge();
    const char* upPage = pageInfo.GetPage();

    if( !upKey->GetActiveRefs() )
    {
        // We've hit a key active reffing us that should be inactive.
        // If it's object is loaded, then it somehow missed getting unloaded.
        // Else it must have missed letting go of us when it got unloaded.
        if( upKey->ObjectIsLoaded() )
        {
            plStatusLog::AddLineS(logFile, "\tHeld by %s [%s] page %s which is loaded but nothing is reffing", 
                upKey->GetName(), 
                plFactory::GetNameOfClass(upKey->GetUoid().GetClassType()),
                upPage);

            return true;
        }
        else
        {
            plStatusLog::AddLineS(logFile, "\tHeld by %s [%s] page %s which isn't even loaded", 
                upKey->GetName(), 
                plFactory::GetNameOfClass(upKey->GetUoid().GetClassType()),
                upPage);

            return true;
        }
    }

    // if the age of this key is different from the age on the baseKey, 
    // we've got a cross age active ref, which is illegal.
    if( stricmp(upAge, baseAge) )
    {
        plStatusLog::AddLineS(logFile, "\tHeld by %s [%s] which is in a different age %s-%s", 
            upKey->GetName(), 
            plFactory::GetNameOfClass(upKey->GetUoid().GetClassType()), 
            upAge,
            upPage);

        return true;
    }

    int numActive = 0;
    int i;
    for( i = 0; i < upKey->GetNumNotifyCreated(); i++ )
    {
        if( upKey->GetActiveBits().IsBitSet(i) )
        {
            numActive++;
        }
    }
    if( numActive < upKey->GetActiveRefs() )
    {
        // Someone has AddRef'd us
        plStatusLog::AddLineS(logFile, "\tHeld by %s [%s] page %s which is loaded due to %d AddRef(s)", 
            upKey->GetName(), 
            plFactory::GetNameOfClass(upKey->GetUoid().GetClassType()), 
            upPage,
            upKey->GetActiveRefs()-numActive);

        return true;
    }

    // Now recurse up the active reference tree.
    for( i = 0; i < upKey->GetNumNotifyCreated(); i++ )
    {
        if( upKey->GetActiveBits().IsBitSet(i) )
        {
            int j;
            for( j = 0; j < upKey->GetNotifyCreated(i)->GetNumReceivers(); j++ )
            {
                plKey reffer = upKey->GetNotifyCreated(i)->GetReceiver(j);
                

                if( IVerifyKeyUnloadedRecur(logFile, baseKey, reffer, baseAge) )
                {
                    return true;
                }
            }
        }
    }
    return false;
}

bool plResManager::VerifyKeyUnloaded(const char* logFile, const plKey& key)
{
    if( key->ObjectIsLoaded() )
    {
        const plPageInfo& pageInfo = FindPage(key->GetUoid().GetLocation())->GetPageInfo();
        const char* age = pageInfo.GetAge();
        const char* page = pageInfo.GetPage();

        plStatusLog::AddLineS(logFile, "==================================");
        plStatusLog::AddLineS(logFile, "Object %s [%s] page %s is loaded", key->GetName(), plFactory::GetNameOfClass(key->GetUoid().GetClassType()), page);

        hsTArray<plKey> tree;
        int cycleStart;
        bool hasCycle = ILookForCyclesRecur(logFile, key, tree, cycleStart);
        if( hasCycle )
        {
            plStatusLog::AddLineS(logFile, "\t%s [%s] held by dependency cycle", key->GetName(), plFactory::GetNameOfClass(key->GetUoid().GetClassType()));
            int i;
            for( i = cycleStart; i < tree.GetCount(); i++ )
            {
                plStatusLog::AddLineS(logFile, "\t%s [%s]", tree[i]->GetName(), plFactory::GetNameOfClass(tree[i]->GetUoid().GetClassType()));
            }
            plStatusLog::AddLineS(logFile, "\tEnd Cycle");
            return true;
        }
        else
        {
            return IVerifyKeyUnloadedRecur(logFile, key, key, age);
        }
    }
    return false;
}

class plValidateKeyIterator : public plRegistryKeyIterator
{
protected:
    plRegistry*     fRegistry;
    const char*     fLogFile;

public:
    plValidateKeyIterator(const char* logFile, plRegistry* reg)
    {
        fRegistry = reg;
        fLogFile = logFile;
    }
    virtual bool EatKey(const plKey& key)
    {
        fRegistry->VerifyKeyUnloaded(fLogFile, key);
        return true;
    }
};

class plValidatePageIterator : public plRegistryPageIterator
{
protected:
    const char*             fAge;
    plRegistryKeyIterator*  fIter;

public:
    plValidatePageIterator(const char* age, plRegistryKeyIterator* iter) : fAge(age), fIter(iter) {}


    virtual bool    EatPage( plRegistryPageNode *keyNode )
    {
        if( !stricmp(fAge, keyNode->GetPageInfo().GetAge()) )
            return keyNode->IterateKeys( fIter );
        return true;
    }
};

void plResManager::VerifyAgeUnloaded(const char* logFile, const char* age)
{
    bool autoLog = false;
    char buff[256];
    if( !logFile || !*logFile )
    {
        sprintf(buff, "%s.log", age);
        logFile = buff;
        autoLog = true;
    }

    if( !autoLog )
    {
        plStatusLog::AddLineS(logFile, "///////////////////////////////////");
        plStatusLog::AddLineS(logFile, "Begin Verification of age %s", age);
    }

    plValidateKeyIterator keyIter(logFile, this);
    plValidatePageIterator pageIter(age, &keyIter);

    IterateAllPages(&pageIter);

    if( !autoLog )
    {
        plStatusLog::AddLineS(logFile, "End Verification of age %s", age);
        plStatusLog::AddLineS(logFile, "///////////////////////////////////");
    }
}
#endif