sysstatemgmt/ssmpolicyplugins/ssmstartuppolicy/src/ssmstartuppolicy.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 02 Feb 2010 00:53:00 +0200
changeset 0 4e1aa6a622a0
permissions -rw-r--r--
Revision: 201003

/*
* Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of "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: Implementation of CSsmStartupPolicy class.
*
*/

#include <centralrepository.h>
#include <e32property.h>
#include <e32uid.h>
#include <featmgr.h>
#include <ssm/ssmdomaindefs.h>
#include <ssm/ssmcmd.hrh>
#include <ssm/ssmcommandlistresourcereader.h>
#include <ssm/ssmstate.h>
#include <ssm/ssmstateawaresession.h>
#include <ssm/ssmstatetransition.h>
#include <ssm/startupdomainpskeys.h>

#include "ssmsubstateext.hrh"
#include "ssmmapperutility.h"
#include "ssmmapperutilitystatic.h"
#include "ssmstartuppolicy.h"
#include "ssmpolicypluginsprivatecrkeys.h"
#include "trace.h"

/**
* Start-up state policy resource file path format :
* "\private\<SID of SSM>\startup\<Value of KSystemStartupModeKey>\"
*/
_LIT( KCommandListPath, "Z:\\private\\2000D75B\\startup\\%d\\" );

// ======== LOCAL FUNCTIONS ========

// ---------------------------------------------------------------------------
// GetStartupMode
// ---------------------------------------------------------------------------
//
static TInt GetStartupMode()
    {
    FUNC_LOG;

    TInt startupMode( EStartupModeNormal );
    TInt errorCode = RProperty::Get(
        SsmMapperUtility::PsUid( KPSUidStartup ), KPSGlobalStartupMode, startupMode );
    ERROR( errorCode, "Failed to read startup mode from P&S" );
    INFO_1( "Startup mode is %d", startupMode );
    return startupMode;
    }


// ---------------------------------------------------------------------------
// DecideStateByStartupMode
// ---------------------------------------------------------------------------
//
static TSsmStartupSubStateExt DecideStateByStartupMode()
    {
    FUNC_LOG;

    TInt startupMode( GetStartupMode() );
    TSsmStartupSubStateExt nextState( ESsmStateSecurityCheck );
    switch ( startupMode )
        {
        case EStartupModeTest:
            nextState = ESsmStateTest;
            break;
        case EStartupModeAlarm:
            nextState = ESsmStateAlarm;
            break;
        case EStartupModeCharging:
            nextState = ESsmStateCharging;
            break;
        default:
            break;
        }

    return nextState;
    }


// ======== MEMBER FUNCTIONS ========

// ---------------------------------------------------------------------------
// CSsmStartupPolicy::NewL
// ---------------------------------------------------------------------------
//
EXPORT_C MSsmStatePolicy* CSsmStartupPolicy::NewL()
    {
    CSsmStartupPolicy* self = new( ELeave ) CSsmStartupPolicy;
    CleanupStack::PushL( self );
    self->ConstructL();
    CleanupStack::Pop( self );
    return self;
    }


// ---------------------------------------------------------------------------
// CSsmStartupPolicy::~CSsmStartupPolicy
// ---------------------------------------------------------------------------
//
CSsmStartupPolicy::~CSsmStartupPolicy()
    {
    FUNC_LOG;
    }


// ---------------------------------------------------------------------------
// CSsmStartupPolicy::GetNextState
// ---------------------------------------------------------------------------
//
TBool CSsmStartupPolicy::GetNextState(
    TSsmState aCurrentTransition,
    TInt /*aReason*/,
    TInt aError,
    TInt aSeverity,
    TSsmState& aNextState )
    {
    FUNC_LOG;
    INFO_2( "Current state %04x.%04x",
        aCurrentTransition.MainState(), aCurrentTransition.SubState() );
    ERROR_1( aError, "State transition resulted in an error, severity %d", aSeverity );
	ASSERT_TRACE( aCurrentTransition.MainState() == ESsmStartup );

	TBool moreToCome = EFalse;
	if ( aError == KErrNone || aSeverity < ECmdHighSeverity )
		{
		switch( aCurrentTransition.SubState() )
			{
			case KSsmAnySubState:
			case ESsmStatePreUiServices:
				aNextState = TSsmState( ESsmStartup, ESsmStateStartingUiServices );
				moreToCome = ETrue;
				break;
			case ESsmStateStartingUiServices:
				aNextState = TSsmState( ESsmStartup, ESsmStateStartingCriticalApps );
				moreToCome = ETrue;
				break;
			case ESsmStateStartingCriticalApps:
				aNextState = TSsmState( ESsmStartup, ESsmStateSelfTestOK );
				moreToCome = ETrue;
				break;
			case ESsmStateSelfTestOK:
				aNextState = TSsmState( ESsmStartup, DecideStateByStartupMode() );
				moreToCome = ETrue;
				break;
			case ESsmStateChargingToNormal:
			case ESsmStateAlarmToNormal:
				aNextState = TSsmState( ESsmStartup, ESsmStateSecurityCheck );
				moreToCome = ETrue;
				break;
			case ESsmStateSecurityCheck:
				aNextState = TSsmState( ESsmStartup, SelectStateAfterSecurityCheck() );
				moreToCome = ETrue;
				break;
			case ESsmStateChargingToAlarm:
				aNextState = TSsmState( ESsmStartup, ESsmStateAlarm );
				moreToCome = ETrue;
				break;
			case ESsmStateAlarmToCharging:
				aNextState = TSsmState( ESsmStartup, ESsmStateCharging );
				moreToCome = ETrue;
				break;
			case ESsmStateNonCritical:
			    ClearResetCounter(); // start-up is successful
				aNextState = TSsmState( ESsmNormal, KSsmAnySubState );
				moreToCome = ETrue;
				break;
			case ESsmStateAlarm:
			case ESsmStateCharging:
			case ESsmStateTest:
				moreToCome = EFalse;
				break;
			case ESsmStateEmergencyCallsOnly:
			    ClearResetCounter(); // start-up is successful
				moreToCome = EFalse;
				break;
			default:
        		ASSERT_ALWAYS_TRACE;
				break;
			}
		}
	else
		{
        if ( ResetLimitReached() ) // Updates the reset count
            {
            aNextState = TSsmState( ESsmFail, KSsmAnySubState );
	    	moreToCome = ETrue;
    	    }
    	else
    	    {
            aNextState = TSsmState( ESsmShutdown, ESsmErrorShutdown );
            moreToCome = ETrue;
    	    }
		}

	return moreToCome;
    }


// ---------------------------------------------------------------------------
// CSsmStartupPolicy::GetCommandListPathL
// ---------------------------------------------------------------------------
//
void CSsmStartupPolicy::GetCommandListPathL( TDes& aCmdListPath )
    {
    FUNC_LOG;

	aCmdListPath.Format( KCommandListPath, iHardwareReason );
	iUtil->GetCommandListPath( aCmdListPath );
    }


// ---------------------------------------------------------------------------
// CSsmStartupPolicy::IsAllowedTargetState
// ---------------------------------------------------------------------------
//
TBool CSsmStartupPolicy::IsAllowedTargetState(
    const TSsmStateTransition& aRequest ) const
    {
    FUNC_LOG;

    TSsmState currentState;
    TInt errorCode = GetCurrentState( currentState );
    ERROR( errorCode, "Failed to get current state" );

    if ( errorCode == KErrNone )
        {
        TUint16 reqMainState( aRequest.State().MainState() );
        TUint16 reqSubState( aRequest.State().SubState() );
        INFO_2( "Requested target state = %04x.%04x", reqMainState, reqSubState );
        TUint16 curMainState( currentState.MainState() );
        TUint16 curSubState( currentState.SubState() );
        INFO_2( "Current state = %04x.%04x", curMainState, curSubState );

        if ( reqMainState == ESsmStartup )
            {
            if ( ( curSubState == ESsmStateCharging &&
                   ( reqSubState == ESsmStateChargingToAlarm ||
                     reqSubState == ESsmStateChargingToNormal ) ) ||
                 ( curSubState == ESsmStateAlarm &&
                   ( reqSubState == ESsmStateAlarmToCharging ||
                     reqSubState == ESsmStateAlarmToNormal ) ) )
                {
                return ETrue;
                }
            }
        else if ( reqMainState == ESsmNormal )
            {
            if ( ( curMainState == ESsmStartup &&
                   curSubState == ESsmStateNonCritical ) )
                {
                return ETrue;
                }
            }
        else if ( reqMainState == ESsmFail ||
                  reqMainState == ESsmShutdown )
            {
            return ETrue;
            }
        }

    return EFalse;
    }


// ---------------------------------------------------------------------------
// CSsmStartupPolicy::TargetSubState
// ---------------------------------------------------------------------------
//
TUint16 CSsmStartupPolicy::TargetSubState( const TUint16 aRequestedSubState )
    {
    FUNC_LOG;
    INFO_1( "Requested sub state %04x", aRequestedSubState );

	return ( aRequestedSubState == KSsmAnySubState ) ?
        ESsmStatePreUiServices : aRequestedSubState;
    }


// ---------------------------------------------------------------------------
// CSsmStartupPolicy::CSsmStartupPolicy
// ---------------------------------------------------------------------------
//
CSsmStartupPolicy::CSsmStartupPolicy()
    {
    FUNC_LOG;
    }


// ---------------------------------------------------------------------------
// CSsmStartupPolicy::ConstructL
// ---------------------------------------------------------------------------
//
void CSsmStartupPolicy::ConstructL()
    {
    FUNC_LOG;

	// Read the hardware reason
	User::LeaveIfError( RProperty::Get(
	    KUidSystemCategory, KSystemStartupModeKey, iHardwareReason ) );

    BaseConstructL();
    }


// ---------------------------------------------------------------------------
// CSsmStartupPolicy::IsSimlessOfflineSupported
// ---------------------------------------------------------------------------
//
TBool CSsmStartupPolicy::IsSimlessOfflineSupported()
    {
    FUNC_LOG;
    TBool ret( EFalse );

    ret = ( iUtil->FeatureStatus( TUid::Uid( KFeatureIdOfflineMode ) ) ||
            iUtil->FeatureStatus( TUid::Uid( KFeatureIdFlightMode ) ) ) &&
            iUtil->FeatureStatus( TUid::Uid( KFeatureIdFfSimlessOfflineSupport ) );

    INFO_1( "Simless offline supported = %d", ret );
    return ret;
    }


// ---------------------------------------------------------------------------
// CSsmStartupPolicy::SelectStateAfterSecurityCheck
// ---------------------------------------------------------------------------
//
TUint16 CSsmStartupPolicy::SelectStateAfterSecurityCheck()
    {
    FUNC_LOG;

    TInt value( ESimStatusUninitialized );
    TInt errorCode = RProperty::Get( iUtil->PsUid( KPSUidStartup ), KPSSimStatus, value );
    ERROR( errorCode, "Failed to read KPSUidStartup::KPSSimStatus P&S key" );
    INFO_1( "Current SIM status = %d", value );
    TUint16 retVal( ESsmStateEmergencyCallsOnly );
    if ( errorCode == KErrNone )
        {
        if ( value == ESimUsable || value == ESimNotSupported )
            {
            retVal = ESsmStateNonCritical;
            }
        else if ( value == ESimNotPresent && IsSimlessOfflineSupported() )
            {
            retVal = ESsmStateNonCritical;
            }
        }

    INFO_1( "State selected after security check = %04x", retVal );
    return retVal;
    }


// ---------------------------------------------------------------------------
// CSsmStartupPolicy::ClearResetCounter
// ---------------------------------------------------------------------------
//
void CSsmStartupPolicy::ClearResetCounter()
    {
    FUNC_LOG;

    TRAPD_ERR( errorCode, ClearResetCounterL() );
    ERROR( errorCode, "Failed to clear reset counter" );
    }


// ---------------------------------------------------------------------------
// ClearResetCounterL
// ---------------------------------------------------------------------------
//
void CSsmStartupPolicy::ClearResetCounterL()
    {
    FUNC_LOG;

    CRepository* repository = CRepository::NewLC( iUtil->CrUid( KCRUidSsmStartupPolicy ) );
    User::LeaveIfError( repository->Set( KSsmStartupErrorResetCounter, 0 ) );
    CleanupStack::PopAndDestroy( repository );
    }