sysstatemgmt/systemstatereferenceplugins/custcmd/src/cmdsimsecuritycheck.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Fri, 12 Mar 2010 15:50:01 +0200
branchRCL_3
changeset 5 1a73e8f1b64d
parent 0 4e1aa6a622a0
child 20 1ddbe54d0645
permissions -rw-r--r--
Revision: 201007 Kit: 201008

// Copyright (c) 2008-2010 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 CStrtSecurityCheckTask class
//




/**
 @file
 @internalComponent
 @released
*/

#include "ssmsecuritychecknotifier.h"
#include "cmdsimsecuritycheck.h"
#include "ssmrefcustomcmdcommon.h"
#include "ssmdebug.h"

#include <e32def.h>
#include <ssm/startupdomainpskeys.h>
#include <ssm/ssmstateawaresession.h>
#include <ssm/ssmpatchableconstants.h>
#include <centralrepository.h>
#include <e32svr.h>
#include <u32hal.h>

//Ordinal postion of the security event observer NewL in the plugin dll
const TInt KSecurityEventObserverNewLOridinal = 1;

//Utility plugin name used for starting security note event observer
_LIT(KSusUtilityPlugin, "ssmclayersup.dll");

/*
 * Function to check whether Invalid Sim feature is enabled or not
 */
TUint32 IsSsmInvalidSimOn()
    {
    TUint32 retVal = 0;

#ifdef __WINS__  
        // KSsmInvalidSim is a ROM patchable constant, so need an emulator equivalent
        // if WINS then read value from epoc.ini, requires licencees to set property in epoc.ini. This value is 0xFFFFFFFF if feature is enabled.
        // Usage: In epoc.ini patchdata_KSsmInvalidSim 0xFFFFFFFF (Can be any non-zero value)
        TUint32 readValueOfInvalidSim = 0;
        if( UserSvr::HalFunction(EHalGroupEmulator, EEmulatorHalIntProperty, (TAny*)"patchdata_KSsmInvalidSim", &readValueOfInvalidSim) == KErrNone )
            {
            retVal = readValueOfInvalidSim;
            }
#else   
        retVal = KSsmInvalidSim;        
        
#endif // __WINS__
    return retVal;
    }

CCustomCmdSimSecurityCheck* CCustomCmdSimSecurityCheck::NewL()
	{
	CCustomCmdSimSecurityCheck* self = new (ELeave) CCustomCmdSimSecurityCheck();
	CleanupStack::PushL(self);
	self->ConstructL();
	CleanupStack::Pop(self);
	return self;
	}

void CCustomCmdSimSecurityCheck::ConstructL()
	{
	User::LeaveIfError(iSsmMiscAdaptation.Connect());
	User::LeaveIfError(iSsmStateManager.Connect());
	User::LeaveIfError(iSsmSusCli.Connect());

	iStartupPSUid = CSsmUiSpecific::StartupPSUid();
    __ASSERT_ALWAYS(0 != iStartupPSUid.iUid, PanicNow(KPanicCustomCmdSimSecurityCheck, EInvalidUid));
	iSecurityStateInfoResult = new (ELeave) TPckg<TSecurityStateInfo>(iSecurityStateInfo);
	iSsmUiSpecific = CSsmUiSpecific::InstanceL();

	iSimStatusPSObserver = CSsmSimStatusPSObserver::NewL(iStartupPSUid, iSsmUiSpecific->SimStatusPropertyKey());
	//Add active object to active scheduler
	CActiveScheduler::Add(this);
	}

CCustomCmdSimSecurityCheck::CCustomCmdSimSecurityCheck()
						: CActive( EPriorityStandard)
	{
	}

CCustomCmdSimSecurityCheck::~CCustomCmdSimSecurityCheck()
	{
	ExecuteCancel();
	iSsmMiscAdaptation.Close();		
	iSsmStateManager.Close();
	iSsmSusCli.Close();

	if(iSsmUiSpecific)
		{
		//Will be deleted after last dereference
		iSsmUiSpecific->Release();
		}
	
	delete iSsmSecurityCheckNotifier;
	delete iSecurityNoteObserver;
	delete iSecurityStateInfoResult;
	delete iSimStatusPSObserver;
	}

TInt CCustomCmdSimSecurityCheck::Initialize(CSsmCustomCommandEnv* /*aCmdEnv*/)
	{
	return KErrNone;
	}

void CCustomCmdSimSecurityCheck::Close()
	{
	}

void CCustomCmdSimSecurityCheck::Release()
	{
	delete this;
	}

void CCustomCmdSimSecurityCheck::Execute(const TDesC8& /*aParams*/, TRequestStatus& aStatus)
	{
	//Load the security event observer
	TSsmSupInfo observerPlugin(KSusUtilityPlugin, KSecurityEventObserverNewLOridinal);
	TInt retLoadSecEventObserver = iSsmSusCli.RequestLoadSup(observerPlugin);

	if(KErrNone != retLoadSecEventObserver)
		{
		DEBUGPRINT2A("Load security event observer SUP failed with error code : %d", retLoadSecEventObserver);
		}
	
	//Set the user request to pending
	aStatus = KRequestPending;
	iExecuteRequest = &aStatus;

	//Start the state change
	StartStateChange(StartupAdaptation::ESIMPresent);
	}

/**
@panic EInternalStateError if the active object is running or the state change is pending
*/
void CCustomCmdSimSecurityCheck::StartStateChange(const TSecurityState aState, const TStrtSecurityNoteType aNoteType)
	{
	__ASSERT_ALWAYS(!IsActive(), PanicNow(KPanicCustomCmdSimSecurityCheck, EInternalStateError));
	__ASSERT_ALWAYS(!iStateChangePending, PanicNow(KPanicCustomCmdSimSecurityCheck, EInternalStateError));

	iSubState = ESecuritySubStateNone;
	iStateChangePending = ETrue;
	iNoteType = aNoteType;
	iState = aState;

	//Request for security state change
	iSsmMiscAdaptation.SecurityStateChange(iState, *iSecurityStateInfoResult, iStatus);
	SetActive();
	}

void CCustomCmdSimSecurityCheck::RunL()
	{
	if(iSubState != ESecuritySubStateNone)
		{
		//Execute sim operations 
		DoExtendedSimOperations();
		return;
		}

	if (iStateChangePending)
		{
		//Start changing the state
		StateChangeL(iStatus.Int());
		}
	else
		{
		if (iStatus == KErrNone)
			{
			//Do further security check
			DoSecurityCheck();
			}
		else
			{
			//Complete client request with result code
			CompleteClientRequest(iStatus.Int());
			}
		}
	}

void CCustomCmdSimSecurityCheck::DoExtendedSimOperations()
	{
    if (KErrNone == iStatus.Int())
        {
        switch (iSubState)
            {
            case ESecuritySubStateSimRemoved:
                DoSimRemoved();
                break;
            case ESecuritySubStateSimNotSupported:
                DoSimNotSupported();
                break;
            case ESecuritySubStateSimReadable:
                DoSimReadable();
                break;
            case ESecuritySubStateSimUsable:
                DoSimUsable();
                break;
            case ESecuritySubStateSimStatusPSObserver:
                DoObserveSimStatusPS();
                break;
            default:
                __ASSERT_DEBUG(ESecuritySubStateNone == iSubState, PanicNow(KPanicCustomCmdSimSecurityCheck, EInvalidSecuritySubState));
                break;
            }
        }
    else
        {
        //Complete client request with result code
        CompleteClientRequest(iStatus.Int());
        }
	}

void CCustomCmdSimSecurityCheck::StateChangeL(const TInt aResultCode)
	{
	__ASSERT_ALWAYS(iStateChangePending, PanicNow(KPanicCustomCmdSimSecurityCheck, EInternalStateError));
	DEBUGPRINT2A("Sim security state change completed with error %d", aResultCode);

	iStateChangePending = EFalse;
	if (aResultCode == KErrNone)
		{
		//Proceed with state change in case previous state completed with out any error
		SecurityCheckL();
		}
	else
		{
		//Complete client request with result code
		CompleteClientRequest(aResultCode);
		}
	}

void CCustomCmdSimSecurityCheck::CompleteClientRequest(TInt aReason)
	{
	//Complete client request with reason code
	if (iExecuteRequest)
        {
        User::RequestComplete(iExecuteRequest, aReason);
        }
	}

void CCustomCmdSimSecurityCheck::SecurityCheckL()
    {
    //Delete the security notifier
	delete iSsmSecurityCheckNotifier;
	iSsmSecurityCheckNotifier = NULL;

	TInt errorCode = KErrNone;
    //Handle the state transition
    switch (iState)
        {
        case StartupAdaptation::ESIMPresent:
			HandleSIMPresent();
            break;
		case StartupAdaptation::ESIMReadable:
            SimReadable();
            break;
		case StartupAdaptation::ESIMInvalid:
		    HandleSIMInvalid();
		    break;
        case StartupAdaptation::ESIMRejected:
        	HandleSIMRejected();
        	break;
        case StartupAdaptation::ESIMBlocked:
        	HandleSIMBlocked();
        	break;
        case StartupAdaptation::EPINRequired:
        	HandlePINRequired();
        	break;
        case StartupAdaptation::ESIMLessOfflineSupported:
            HandleSIMLessOfflineSupported();
            break;
		case StartupAdaptation::ESIMCodesOK:
			StartStateChange( StartupAdaptation::ESIMLock );
            break;
        case StartupAdaptation::ESecurityCheckOK:
        	errorCode = RProperty::Set(CSsmUiSpecific::StarterPSUid(), KStarterSecurityPhase, EStarterSecurityPhaseSimOk );
        	if(KErrNone != errorCode)
        		{
        		DEBUGPRINT2A("Failed to set value of property CSsmUiSpecific::StarterPSUid() because of error %d", errorCode);
        		}
        	CompleteClientRequest(KErrNone);
        	break;
        case StartupAdaptation::ESecurityCheckFailed:
        	errorCode = RProperty::Set(CSsmUiSpecific::StarterPSUid(), KStarterSecurityPhase, EStarterSecurityPhaseSimNok);
        	if(KErrNone != errorCode)
        		{
        		DEBUGPRINT2A("Failed to set value of property CSsmUiSpecific::StarterPSUid() because of error %d", errorCode);
        		}
        	CompleteClientRequest(KErrNone);
        	break;
        case StartupAdaptation::ESIMLock:
            if (iSecurityStateInfo == StartupAdaptation::ESimLockRestrictionOn)
                {
                StartSecurityNoteObserverL();
                iSsmSecurityCheckNotifier = CSsmSecurityCheckNotifier::NewL(ESecInfoSimLockRestrOn);
                iSsmSecurityCheckNotifier->ShowNoteL(iStatus);
                SetActive();
                }
            else
                {
				DoSecurityCheck();
                }
            break;
        case StartupAdaptation::EAskPIN:
        case StartupAdaptation::EAskPUK:
            StartSecurityNoteObserverL();
			iSsmSecurityCheckNotifier = CSsmSecurityCheckNotifier::NewL(iNoteType);
            iSsmSecurityCheckNotifier->ShowNoteL(iStatus);
            SetActive();
			break;
        default:
            CompleteClientRequest(KErrNotSupported);
            break;
        };
    }

void CCustomCmdSimSecurityCheck::DoSecurityCheck()
    {
    //Handle pending state transition
    switch (iState)
        {
        case StartupAdaptation::ESIMLock:
            HandleSIMLock();
            break;
        case StartupAdaptation::EAskPIN:
            HandlePINAsked();
            break;
        case StartupAdaptation::EAskPUK:
            HandlePUKAsked();
            break;
        default:
            CompleteClientRequest(KErrNotSupported);
            break;
        };
	}

void CCustomCmdSimSecurityCheck::HandleSIMPresent()
    {
    switch (iSecurityStateInfo)
    	{
    	case StartupAdaptation::EYes:
    		//Start sim readable state transition
    		StartStateChange(StartupAdaptation::ESIMReadable);
    		break;
    	case StartupAdaptation::ENo:
    		if (IsSimSupported())
    			{
    			SimRemoved();
    			}
    		else
    			{
    			SimNotSupported();
    			}
    		break;
    	default:
    		DEBUGPRINT2A("Illegal security result in ESIMPresent state: %d", iSecurityStateInfo);
    		CompleteClientRequest(KErrGeneral);
    		break;
    	};
    }

void CCustomCmdSimSecurityCheck::DoSimRemoved()
	{
	//Start sim less offline support state transition
	StartStateChange(StartupAdaptation::ESIMLessOfflineSupported);
	}

void CCustomCmdSimSecurityCheck::DoSimNotSupported()
	{
	//Start sim less offline support state transition
	StartStateChange(StartupAdaptation::ESIMLessOfflineSupported);
	}

void CCustomCmdSimSecurityCheck::DoSimReadable()
	{
	if(IsSsmInvalidSimOn())
	    {
	    //Start sim invalid state transition
	    StartStateChange(StartupAdaptation::ESIMInvalid);
	    }
	else
	    {
	    //Start sim rejected state transition
        StartStateChange(StartupAdaptation::ESIMRejected);
	    }
	}

void CCustomCmdSimSecurityCheck::DoSimUsable()
	{
	TInt simStatus = ESimStatusUninitialized;
	TInt errVal = (iSimStatusPSObserver->GetValue(simStatus));
	
	if (KErrNone == errVal)
	    {
	    //Do not observe if SIM status has already changed to ESimUsable
	     if(simStatus == ESimUsable)
	         {
	         //Start security check ok state transition
	         StartStateChange(StartupAdaptation::ESecurityCheckOK);
	         }
	     else
	         {
	         //Wait till the SIM status changes to ESimUsable. 
	         //Start observing for the P&S key change
	         iSimStatusPSObserver->StartObserving(iStatus);
	         iSubState = ESecuritySubStateSimStatusPSObserver;
	         SetActive();
	         }
	    }
	else
	    {
	    DEBUGPRINT2A("Error when getting SIM status value: %d", errVal);
	    CompleteClientRequest(errVal);
	    }
	}

void CCustomCmdSimSecurityCheck::DoObserveSimStatusPS()
    {
    //Start security check ok state transition
    StartStateChange(StartupAdaptation::ESecurityCheckOK);
    }

void CCustomCmdSimSecurityCheck::SimRemoved()
    {
    TUint simStatusPropertyKey = iSsmUiSpecific->SimStatusPropertyKey();
    TSsmSwp swp(simStatusPropertyKey, ESimNotPresent);
    iSsmStateManager.RequestSwpChange(swp, iStatus);
	iSubState = ESecuritySubStateSimRemoved;
	SetActive();
    }

void CCustomCmdSimSecurityCheck::SimNotSupported()
	{
    TUint simStatusPropertyKey = iSsmUiSpecific->SimStatusPropertyKey();
    TSsmSwp swp(simStatusPropertyKey, ESimNotSupported);
	iSsmStateManager.RequestSwpChange(swp, iStatus);
	iSubState = ESecuritySubStateSimNotSupported;
	SetActive();
	}

void CCustomCmdSimSecurityCheck::SimReadable()
	{
    TUint simStatusPropertyKey = iSsmUiSpecific->SimStatusPropertyKey();
    TSsmSwp swp(simStatusPropertyKey, ESimReadable);
	iSsmStateManager.RequestSwpChange(swp, iStatus);
	iSubState = ESecuritySubStateSimReadable;
	SetActive();
	}

void CCustomCmdSimSecurityCheck::SimUsable()
	{
	TUint simStatusPropertyKey = iSsmUiSpecific->SimStatusPropertyKey();
    TSsmSwp swp(simStatusPropertyKey, ESimUsable);
    iSsmStateManager.RequestSwpChange(swp, iStatus);
	iSubState = ESecuritySubStateSimUsable;
	SetActive();	
 	}

void CCustomCmdSimSecurityCheck::HandleSIMInvalid()
    {
    switch (iSecurityStateInfo)
        {
        case StartupAdaptation::EYes:

            //Publish the Sim Security Status as ESimInvalid
            RProperty::Set(iStartupPSUid, KStartupSimSecurityStatus, ESimInvalid);
            
            //Set the security state as failed
            SetSecurityStatus(EStrtSecurityCheckFailed);

            //Start sim less offline support state transition
            StartStateChange(StartupAdaptation::ESecurityCheckFailed);
            break;
        case StartupAdaptation::ENo:
            //Start sim rejected state transition
            StartStateChange(StartupAdaptation::ESIMRejected);
            break;
        default:
            DEBUGPRINT2A("Illegal security result in ESIMInvalid state: %d", iSecurityStateInfo);
            //Complete user request with KErrGeneral
            CompleteClientRequest(KErrGeneral);
            break;
        };
    }


void CCustomCmdSimSecurityCheck::HandleSIMRejected()
    {
    switch (iSecurityStateInfo)
    	{
    	case StartupAdaptation::EYes:
    		RProperty::Set(iStartupPSUid, KStartupSimSecurityStatus, ESimRejected );

    		//Set the security state as failed
			SetSecurityStatus(EStrtSecurityCheckFailed);

			//Start sim less offline support state transition
			StartStateChange(StartupAdaptation::ESecurityCheckFailed);
			break;
    	case StartupAdaptation::ENo:
			//Start sim blocked state transition
			StartStateChange(StartupAdaptation::ESIMBlocked);
			break;
    	default:
    		DEBUGPRINT2A("Illegal security result in ESIMRejected state: %d", iSecurityStateInfo);

    		//Complete user request with KErrGeneral
    		CompleteClientRequest(KErrGeneral);
    		break;
    	};
    }

void CCustomCmdSimSecurityCheck::HandleSIMBlocked()
    {
    switch (iSecurityStateInfo)
    	{
    	case StartupAdaptation::EPUK1Required:
    		//Start PUK required state transition
    		StartStateChange(StartupAdaptation::EAskPUK, ESecCodePUK1);
    		break;
    	case StartupAdaptation::EUPUKRequired:
    		//Start UPUK required state transition
    		StartStateChange(StartupAdaptation::EAskPUK, ESecCodeUPUK);
    		break;
    	case StartupAdaptation::ENo:
    		//Start PIN required state transition
    		StartStateChange(StartupAdaptation::EPINRequired);
    		break;
    	default:
    		DEBUGPRINT2A("Illegal security result in ESIMBlocked state: %d", iSecurityStateInfo);

    		//Complete user request with KErrGeneral
    		CompleteClientRequest(KErrGeneral);
    		break;
    	};
    }

void CCustomCmdSimSecurityCheck::HandlePINRequired()
    {
    switch (iSecurityStateInfo)
    	{
    	case StartupAdaptation::EPIN1Required:
    		//Start PIN required state transition
    		StartStateChange(StartupAdaptation::EAskPIN, ESecCodePIN1);
    		break;
    	case StartupAdaptation::EUPINRequired:
    		//Start UPIN required state transition
    		StartStateChange(StartupAdaptation::EAskPIN, ESecCodeUPIN);
    		break;
    	case StartupAdaptation::ENo:
    		//Start SIM Codes OK state transition
    		StartStateChange(StartupAdaptation::ESIMCodesOK);
    		break;
    	default:
    		DEBUGPRINT2A( "Illegal security result in EPINRequired state: %d", iSecurityStateInfo );

    		//Complete user request with KErrGeneral
    		CompleteClientRequest(KErrGeneral);
    		break;
    	};
    }

void CCustomCmdSimSecurityCheck::HandleSIMLessOfflineSupported()
    {
    if (IsSimSupported())
        {
        if (iSsmUiSpecific->IsSimlessOfflineSupported())
            {
            //SIMless offline is supported

            //Set the security state as Sim less offline
            SetSecurityStatus(EStrtSimlessOffline);
            StartStateChange(StartupAdaptation::ESIMLock);
            }
        else
            {
            //SIMless offline is not supported

            //Set the security state as failed
            SetSecurityStatus(EStrtSecurityCheckFailed);
            StartStateChange(StartupAdaptation::ESecurityCheckFailed);
            }
        }
    else
        {
        //SIM not supported
        SetSecurityStatus(EStrtSimlessOffline);
        StartStateChange(StartupAdaptation::ESIMLock);
        }
    }

TBool CCustomCmdSimSecurityCheck::IsSimSupported()
	{
	return iSsmUiSpecific->IsSimSupported();
	}

void CCustomCmdSimSecurityCheck::HandleSIMLock()
    {
    switch(iSecurityStateInfo)
    	{
    	case StartupAdaptation::ESimLockRestricted:
    		//Set the sim lock status property to "Sim lock active"
		    RProperty::Set(iStartupPSUid, KStartupSimLockStatus, ESimLockActive );

		    //Set the sim security status property to "Sim unaccepted"
		    RProperty::Set(iStartupPSUid, KStartupSimSecurityStatus, ESimUnaccepted );

	        //Set the security state as failed
	        SetSecurityStatus(EStrtSecurityCheckFailed);

	        //Start security check failed state transition
	        StartStateChange(StartupAdaptation::ESecurityCheckFailed);
			break;
		case StartupAdaptation::ESimLockOk:
	       //Set the sim lock status property to "Sim lock ok"
	        RProperty::Set(iStartupPSUid, KStartupSimLockStatus, ESimLockOk );
	        SetSecurityCheckOK();
			break;
		case StartupAdaptation::ESimLockRestrictionOn:
			//Set the sim lock status property to "Sim lock restriction on"
	        RProperty::Set(iStartupPSUid, KStartupSimLockStatus, ESimLockRestrictionOn);
	        SetSecurityCheckOK();
			break;
		case StartupAdaptation::ESimLockRestrictionPending:
	        //Set the sim lock status property to "Sim lock restriction pending"
	        RProperty::Set(iStartupPSUid, KStartupSimLockStatus, ESimLockRestrictionPending);

	        //Set the sim security status property to "Sim unaccepted"
	        RProperty::Set(iStartupPSUid, KStartupSimSecurityStatus, ESimUnaccepted);

	        //Set the security state as failed
	        SetSecurityStatus( EStrtSecurityCheckFailed);

	        //Start security check failed state transition
	        StartStateChange(StartupAdaptation::ESecurityCheckFailed);
			break;
		default:
			DEBUGPRINT2A("Illegal security result in ESIMLock state: %d", iSecurityStateInfo);
			//Complete user request with KErrGeneral
			CompleteClientRequest(KErrGeneral);
			break;
		}

    }

void CCustomCmdSimSecurityCheck::SetSecurityCheckOK()
    {
    if (SecurityStatus() != EStrtSimlessOffline)
        {
        //Set the security state as passed
        SetSecurityStatus( EStrtSecurityCheckPassed);
        SimUsable();
        }
	else
		{
		//Start security check ok state transition
    	StartStateChange(StartupAdaptation::ESecurityCheckOK);
		}
    }


void CCustomCmdSimSecurityCheck::HandlePINAsked()
    {
	__ASSERT_DEBUG(NULL != iSsmSecurityCheckNotifier, PanicNow(KPanicCustomCmdSimSecurityCheck, ECmdNullPtr));

    const TBool isCodeAccepted = iSsmSecurityCheckNotifier->IsCodeAccepted();

    //Try again until PIN is either accepted or blocked.
    if (isCodeAccepted)
        {
        //Start SIM codes ok state transition
        StartStateChange(StartupAdaptation::ESIMCodesOK);
        }
    else
        {
        switch(iNoteType)
      		{
      		case ESecCodePIN1:
      			//Start security code PUK1 state transition
				StartStateChange(StartupAdaptation::EAskPUK, ESecCodePUK1);
				break;
      		case ESecCodeUPIN:
      			//Start security code UPUK state transition
      			StartStateChange(StartupAdaptation::EAskPUK, ESecCodeUPUK);
      			break;
      		default:
                DEBUGPRINT2A("Illegal code type in EAskPIN state: %d", iNoteType);
                //Complete user request with KErrGeneral
                CompleteClientRequest(KErrGeneral);
      			break;
      		}
        }
    }

void CCustomCmdSimSecurityCheck::HandlePUKAsked()
    {
	__ASSERT_DEBUG(NULL != iSsmSecurityCheckNotifier, PanicNow(KPanicCustomCmdSimSecurityCheck, ECmdNullPtr));
    
	//Is the entered code correct
    const TBool isCodeAccepted = iSsmSecurityCheckNotifier->IsCodeAccepted();

    if (isCodeAccepted)
        {
        //Start SIM code ok state transition
        StartStateChange(StartupAdaptation::ESIMCodesOK);
        }
    else
        {
        //Set the Sim security status property to "Sim rejected"
        RProperty::Set(iStartupPSUid, KStartupSimSecurityStatus, ESimRejected);

        //Set the security state as failed
        SetSecurityStatus(EStrtSecurityCheckFailed);

        //Start security check failed state transition
        StartStateChange(StartupAdaptation::ESecurityCheckFailed);
        }
    }

void CCustomCmdSimSecurityCheck::SetSecurityStatus(const TStrtSecurityStatus& aSecurityStatus)
    {
    iSsmUiSpecific->SetSecurityStatus(aSecurityStatus);
    }

TStrtSecurityStatus CCustomCmdSimSecurityCheck::SecurityStatus() const
    {
    return iSsmUiSpecific->SecurityStatus();
    }

void CCustomCmdSimSecurityCheck::StartSecurityNoteObserverL()
	{
	if(!iSecurityNoteObserver)
		{
		iSecurityNoteObserver = CSecurityNoteObserver::NewL();
		}
	}

TInt CCustomCmdSimSecurityCheck::RunError(TInt aError)
	{
	//Do the final cleanup incase of error
	Cleanup(aError);
	return KErrNone;
	}

void CCustomCmdSimSecurityCheck::DoCancel()
	{
	// Cancel all possible requests that could be outstanding on iStatus
	if(iSsmSecurityCheckNotifier)
		{
		iSsmSecurityCheckNotifier->Cancel();
		}
	if(iSecurityNoteObserver)
		{
		iSecurityNoteObserver->Cancel();
		}
	iSsmMiscAdaptation.Cancel();
	iSsmStateManager.RequestSwpChangeCancel();
	iSimStatusPSObserver->Cancel();
	}

void CCustomCmdSimSecurityCheck::ExecuteCancel()
	{
	//Call cancel method of CActive	
	Cancel();

	//Do the final cleanup before cancelling
	Cleanup(KErrCancel);
	}

void CCustomCmdSimSecurityCheck::Cleanup(TInt aError)
	{
	if (iExecuteRequest)
		{
		//Complete the user request with the given error code
		User::RequestComplete(iExecuteRequest, aError);
		}
	}

// CSsmSimStatusPSObserver 
// Active Object Class that observes for SIM status P&S key change
CSsmSimStatusPSObserver* CSsmSimStatusPSObserver::NewL(TUid aCategory, TUint aKey)
	{
	CSsmSimStatusPSObserver* self = new (ELeave) CSsmSimStatusPSObserver(aCategory, aKey);
	CleanupStack::PushL(self);
	self->ConstructL();
	CleanupStack::Pop(self);
	return self;
	}

void CSsmSimStatusPSObserver::ConstructL()
	{
	User::LeaveIfError(iProperty.Attach(iCategory, iKey));
	}

CSsmSimStatusPSObserver::CSsmSimStatusPSObserver(TUid aCategory, TUint aKey)  
                                    : CActive(EPriorityStandard), iCategory(aCategory), iKey(aKey)
    {
    CActiveScheduler::Add(this);
    }

CSsmSimStatusPSObserver::~CSsmSimStatusPSObserver()
    {
    Cancel();
    iProperty.Close();
    }

void CSsmSimStatusPSObserver::StartObserving(TRequestStatus& aStatus)
    {
    __ASSERT_DEBUG(!(IsActive()), PanicNow(KPanicCustomCmdSimSecurityCheck, EInternalStateError));
    aStatus = KRequestPending;
    iClientStatus = &aStatus;
    iProperty.Subscribe(iStatus);
    SetActive();
    }

TInt CSsmSimStatusPSObserver::GetValue(TInt& aValue)
    {
    TInt errVal = (iProperty.Get(aValue));
    return errVal;
    }

void CSsmSimStatusPSObserver::DoCancel()
    {
    iProperty.Cancel();
    Complete(KErrCancel);
    }

void CSsmSimStatusPSObserver::RunL()
    {
    Complete(iStatus.Int());
    }

void CSsmSimStatusPSObserver::Complete(const TInt aErrorCode)
    {
    if (iClientStatus)
        {
        User::RequestComplete(iClientStatus, aErrorCode);
        iClientStatus = NULL;
        }
    }