webengine/osswebengine/MemoryManager/Src/StopScheduler.cpp
author Simon Howkins <simonh@symbian.org>
Mon, 15 Nov 2010 14:53:34 +0000
branchRCL_3
changeset 105 871af676edac
parent 94 919f36ff910f
permissions -rw-r--r--
Adjusted to avoid exports, etc, from a top-level bld.inf

/*
* Copyright (c) 2006 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of the License "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:  
*
*
*/
#include "StopScheduler.h"
#include "MemoryPool.h"
#include "fast_malloc.h"
#include "MemoryLogger.h"
#include <hal.h>

// MEMBER FUNCTIONS
static const TTimeIntervalMicroSeconds32 KMemoryCheckIntervalSlow = 1000000;        // 1 second
static const TTimeIntervalMicroSeconds32 KMemoryCheckIntervalFast = 500000;         // 0.5 second

//-----------------------------------------------------------------------------
// CStopScheduler::CStopScheduler
//-----------------------------------------------------------------------------
CStopScheduler::CStopScheduler(CMemoryPool& aMemoryPool)
  : CActive( CActive::EPriorityHigh ), iMemoryPool( aMemoryPool ), iState( EIdle ), iNextStop( EAllStop )
    {
    CActiveScheduler::Add( this );
    iCheckTimer.CreateLocal();
    }

//-----------------------------------------------------------------------------
// CStopScheduler::~CStopScheduler
//-----------------------------------------------------------------------------
CStopScheduler::~CStopScheduler()
    {
    iCheckTimer.Close();
    Cancel();
    }

//-----------------------------------------------------------------------------
// CStopScheduler::Start
//-----------------------------------------------------------------------------
void CStopScheduler::Start( TSchedulerState aState, TInt aMemSize )
    {
    // stopping has a higher priority than collecting
    if( aState <= iState ) return;

    Cancel();

    iMemRequested = aMemSize;
    iState = aState;
    SelfComplete();

    iNextStop = EAllStop;
    }

void CStopScheduler::SelfComplete()
    {
    TRequestStatus* status = &iStatus;
    SetActive();
    User::RequestComplete( status, iState );    
    }

//-----------------------------------------------------------------------------
// CStopScheduler::RunL
//-----------------------------------------------------------------------------
void CStopScheduler::RunL()
    {
    if( iState == EStopLoading )
        {
        StopLoading( EOOM_PriorityLow );
        iNextStop &= ~ENormalStop;
        CheckMemoryDefered( KMemoryCheckIntervalFast );
        }
    else if( iState == ECheckMemory )
        {
    
        TInt systemFreeRAM; 
        if(HAL::Get(HALData::EMemoryRAMFree, systemFreeRAM) != KErrNone) 
            return;        
        
        // case 1: Good memory levels - no worry
        if( systemFreeRAM > KGoodMemoryThreshold ) // Above 4MB
            {
            iMemoryPool.RestoreCollectors( EOOM_PriorityLow ); // restore all collectors
            iState = EIdle;
            iNextStop = EAllStop;

            // recover the rescue buffer, if it is already released.
            iMemoryPool.RestoreRescueBuffer();
            return;
            }
        
        // case 2: Going low - keep checking
        if( systemFreeRAM > KLowMemoryThreshold ) // Between 4MB to 2MB
            {
            CheckMemoryDefered( KMemoryCheckIntervalSlow );
            iNextStop = EAllStop;
            return;
            }
         
        // case 3: Below normal levels - stop low priority activites
        if( systemFreeRAM > KStopThreshold )  // Between 2MB to 1MB
            {
            if( iNextStop & ENormalStop ) 
                {
                StopLoading( EOOM_PriorityLow );
                iNextStop &= ~ENormalStop;
                }
            CheckMemoryDefered( KMemoryCheckIntervalFast );
            }        
        else // case 4: Really low - stop in emergency (all activites), below 1MB
            {
            if( iNextStop & EEmergencyStop ) 
                {
                StopLoading( EOOM_PriorityHigh );
                iNextStop &= ~EEmergencyStop;
                }
            CheckMemoryDefered( KMemoryCheckIntervalSlow ); // need to restore colectors later
            }
        }
    }

//-----------------------------------------------------------------------------
// CStopScheduler::DoCancel
//-----------------------------------------------------------------------------
void CStopScheduler::DoCancel()
    {
    iState = EIdle;
    }

//-----------------------------------------------------------------------------
// CStopScheduler::StopLoading
//-----------------------------------------------------------------------------
void CStopScheduler::StopLoading( TOOMPriority aPriority )
    {
    MEM_LOG("CStopScheduler::StopLoading - run");
    // stop operations
    for( TInt i=0; i<iMemoryPool.Stoppers().Count(); ++i )
        {
        MOOMStopper* stopper = iMemoryPool.Stoppers()[i];
        if( stopper->Priority() == aPriority )
            stopper->Stop();
        }
    iMemoryPool.CollectMemory(2*KGoodMemoryThreshold);  // try to collect memory
    iMemoryPool.SetStopping( EFalse );
    }

//-----------------------------------------------------------------------------
// CStopScheduler::CheckMemoryDefered
//-----------------------------------------------------------------------------
void CStopScheduler::CheckMemoryDefered( TTimeIntervalMicroSeconds32 aDelay )
    {
    iState = ECheckMemory;
    if( !IsActive() )
        {
        iCheckTimer.After( iStatus, aDelay );
        SetActive();
        }
    }
// END OF FILE