os/persistentdata/featuremgmt/featuremgr/src/serverexe/burstate.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 /**
    17  @file
    18 */
    19 
    20 #include "burstate.h"
    21 #include <babackup.h>
    22 #include "featmgrdebug.h"
    23 #include "featmgrserver.h"
    24 
    25 CBurState::CBurState()
    26 	{
    27 	return;
    28 	}
    29 
    30 /**
    31  * The Backup and Restore state machine.
    32  * This state machine has two state tables. The first table defines the correct modes of operation
    33  * and the second describe the pathways arrising from error conditions. By seperating the normal
    34  * pathways from the error pathways we can simplify the state machine into two parts. The function 
    35  * SetUpBurStruct() will define the pathways for both these tables.
    36  * Each table has three items per row: a current state, an ending state, and a pointer to a function 
    37  * that transforms between these two states safely. 
    38  * Given that we already know where we are, and where we are told to goto, the state machine
    39  * function BUR_StateMachine() will search the normal operation table for a match, and then apply
    40  * the function through the pointer. If no match is found, then the goto state is set to Error, and
    41  * the error table is searched and applied. Because all the state are covered within the error table
    42  * we can always return to a normal more of operation through the application of the error
    43  * functions.
    44  */ 
    45 CBurState::CBurState(CFeatMgrServer* aServer) : iCurrentBURStatus( EFeatMgrBURState_None ), iServer( aServer )
    46 	{
    47 	SetUpBurStruct();
    48 	return;
    49 	}
    50 
    51 CBurState::~CBurState()
    52 	{
    53 	return;
    54 	}
    55 
    56 CBurState* CBurState::NewLC()
    57 	{
    58 	CBurState* self = new (ELeave)CBurState();
    59 	CleanupStack::PushL(self);
    60 	self->ConstructL();
    61 	return self;
    62 	}
    63 
    64 CBurState* CBurState::NewL()
    65 	{
    66 	CBurState* self=CBurState::NewLC();
    67 	CleanupStack::Pop(); // self;
    68 	return self;
    69 	}
    70 
    71 void CBurState::ConstructL()
    72 	{
    73 	return;
    74 	}
    75 
    76 /**
    77  * This function checks the arguments that are passed into HandleBackupOperationEventL. 
    78  * The TBackupOperationAttributes are checked against known values and an error is returned if
    79  * the Feature Manager doesn't recognise the events.
    80  */ 
    81 TInt CBurState::CheckBURArguments(  const TBackupOperationAttributes& aBackupOperationAttributes )
    82 	{
    83 	TInt error( KErrNone );
    84     iBURLockFlag = aBackupOperationAttributes.iFileFlag;
    85     iBUROpType   = aBackupOperationAttributes.iOperation;
    86     
    87 	// determine the operation type (backup or restore)
    88 	switch( iBURLockFlag )	
    89 		{
    90 		case MBackupObserver::EReleaseLockReadOnly:
    91 			INFO_LOG( "MBackupObserver::EReleaseLockReadOnly" );
    92 			break;
    93 		case MBackupObserver::EReleaseLockNoAccess:
    94 			INFO_LOG( "MBackupObserver::EReleaseLockNoAccess" );
    95 			break;
    96 		case MBackupObserver::ETakeLock:
    97 			INFO_LOG( "MBackupObserver::ETakeLock" );
    98 			break;
    99 		default:
   100 			INFO_LOG( "CheckBURArguments iBURLockFlag default" );
   101 			error = KErrNotFound;
   102 			break;
   103 		}
   104 		
   105 	// determine the operation status (e.g. starting, ending)
   106 	switch( iBUROpType )
   107 		{
   108 		case MBackupOperationObserver::ENone:
   109 			INFO_LOG( "ENone" );
   110 			break;
   111 		case MBackupOperationObserver::EStart:
   112 			INFO_LOG( "EStart" );
   113 			break;
   114 		case MBackupOperationObserver::EEnd:
   115 			INFO_LOG( "EEnd" );
   116 			break;
   117 		case MBackupOperationObserver::EAbort:
   118 			INFO_LOG( "EAbort" );	
   119 			break;
   120 		default:
   121 			INFO_LOG( "CheckBURArguments iOperation default" );	
   122 			error = KErrNotFound;
   123 			break;
   124 		}
   125 
   126 	return error;
   127 	}
   128 
   129 /**
   130  * Sets up the array entries for the possible backup/restore state vectors. 
   131  *               from           -> to             : function
   132  * State vector: None           -> BackupStarted  : Goto_StartBackupState
   133  * State vector: BackupStarted  -> BackupEnded    : EndBackupState
   134  * State vector: BackupEnded    -> None           : Goto_NormalState
   135  * State vector: None           -> RestoreStarted : Goto_StartRestoreState
   136  * State vector: RestoreStarted -> RestoreEnded   : Goto_EndRestoreState
   137  * State vector: RestoreEnded   -> None           : Goto_NormalState
   138  */
   139 void CBurState::SetUpBurStruct()
   140 	{
   141 	BURStruct bStruct;
   142 
   143 	// Backup states
   144 
   145 	// State vector: None -> BackupStarted : Goto_StartBackupState
   146 	bStruct.iCurrent = EFeatMgrBURState_None;
   147 	bStruct.iGoto = EFeatMgrBURState_BackupStarted;
   148 	bStruct.iFunc = &CFeatMgrServer::Goto_StartBackupState;
   149 	iBURStructArray[0] = bStruct;
   150 
   151 	// State vector: BackupStarted -> BackupEnded : EndBackupState
   152 	bStruct.iCurrent = EFeatMgrBURState_BackupStarted;
   153 	bStruct.iGoto = EFeatMgrBURState_BackupEnded;
   154 	bStruct.iFunc = &CFeatMgrServer::Goto_EndBackupState;
   155 	iBURStructArray[1] = bStruct;
   156 
   157 	// State vector: BackupEnded -> None : Goto_NormalState
   158 	bStruct.iCurrent = EFeatMgrBURState_BackupEnded;
   159 	bStruct.iGoto = EFeatMgrBURState_None;
   160 	bStruct.iFunc = &CFeatMgrServer::Goto_NormalState;
   161 	iBURStructArray[2] = bStruct;
   162 
   163 	// Valid restore states
   164 	
   165 	// State vector: None -> RestoreStarted : Goto_StartRestoreState
   166 	bStruct.iCurrent = EFeatMgrBURState_None;
   167 	bStruct.iGoto = EFeatMgrBURState_RestoreStarted;
   168 	bStruct.iFunc = &CFeatMgrServer::Goto_StartRestoreState;
   169 	iBURStructArray[3] = bStruct;
   170 
   171 	// State vector: RestoreStarted -> RestoreEnded : Goto_EndRestoreState
   172 	bStruct.iCurrent = EFeatMgrBURState_RestoreStarted;
   173 	bStruct.iGoto = EFeatMgrBURState_RestoreEnded;
   174 	bStruct.iFunc = &CFeatMgrServer::Goto_EndRestoreState;
   175 	iBURStructArray[4] = bStruct;
   176 
   177 	// State vector: RestoreEnded -> None : Goto_NormalState
   178 	bStruct.iCurrent = EFeatMgrBURState_RestoreEnded;
   179 	bStruct.iGoto = EFeatMgrBURState_None;
   180 	bStruct.iFunc = &CFeatMgrServer::Goto_NormalState;
   181 	iBURStructArray[6] = bStruct;
   182 
   183 	// State vector: None -> None : Goto_NormalState
   184 	bStruct.iCurrent = EFeatMgrBURState_None;
   185 	bStruct.iGoto = EFeatMgrBURState_None;
   186 	bStruct.iFunc = &CFeatMgrServer::Goto_NormalState;
   187 	iBURStructArray[5] = bStruct;
   188 
   189 	
   190 	//
   191 	// Error states
   192 
   193 	// State vector: RestoreStarted -> Error : Goto_ErrorState
   194 	bStruct.iCurrent = EFeatMgrBURState_RestoreStarted;
   195 	bStruct.iGoto = EFeatMgrBURState_Error;
   196 	bStruct.iFunc = &CFeatMgrServer::Goto_ErrorState;
   197 	iBURStructErrorArray[0] = bStruct;
   198 
   199 	// State vector: RestoreEnded -> Error : Goto_ErrorState
   200 	bStruct.iCurrent = EFeatMgrBURState_RestoreEnded;
   201 	bStruct.iGoto = EFeatMgrBURState_Error;
   202 	bStruct.iFunc = &CFeatMgrServer::Goto_ErrorState;
   203 	iBURStructErrorArray[1] = bStruct;
   204 
   205 	// State vector: BackupStarted -> None : Goto_ErrorState
   206 	bStruct.iCurrent = EFeatMgrBURState_BackupStarted;
   207 	bStruct.iGoto = EFeatMgrBURState_Error;
   208 	bStruct.iFunc = &CFeatMgrServer::Goto_ErrorState;
   209 	iBURStructErrorArray[2] = bStruct;
   210 	
   211 	// State vector: BackupEnded -> None : Goto_ErrorState
   212 	bStruct.iCurrent = EFeatMgrBURState_BackupEnded;
   213 	bStruct.iGoto = EFeatMgrBURState_Error;
   214 	bStruct.iFunc = &CFeatMgrServer::Goto_ErrorState;
   215 	iBURStructErrorArray[3] = bStruct;
   216 	
   217 	// State vector: Error -> None : Goto_NormalState
   218 	bStruct.iCurrent = EFeatMgrBURState_Error;
   219 	bStruct.iGoto = EFeatMgrBURState_None;
   220 	bStruct.iFunc = &CFeatMgrServer::Goto_NormalState;
   221 	iBURStructErrorArray[4] = bStruct;
   222 	
   223 	return;
   224 	}
   225 
   226 /**
   227  * Convert from the type provided in TBackupOperationAttributes into more sensible BUR types that
   228  * can be handles by the BUR_StateMachine. This step is necessary for two reasons, the first is
   229  * simplification and the second is a lexical check for invalid type MBackupObserver::TFileLockFlags.
   230  */
   231 BURStatus CBurState::ConvertToBURState(  const TBackupOperationAttributes& aBackupOperationAttributes )
   232 	{
   233 	BURStatus status = EFeatMgrBURState_Error;
   234     iBURLockFlag = aBackupOperationAttributes.iFileFlag;
   235     iBUROpType   = aBackupOperationAttributes.iOperation;
   236 	// we ignore the iOperation state
   237     
   238 	if( iBURLockFlag == MBackupObserver::ETakeLock )
   239 		{
   240 		// ending (this doesn't define whether it was a backup
   241 		// or a restore ending, just that file writing can now start 
   242 		// again).
   243 		INFO_LOG("ChangeFileLockL() BUR ending");
   244 		switch( iCurrentBURStatus )
   245 			{
   246 			case( EFeatMgrBURState_BackupStarted ):
   247 				status = EFeatMgrBURState_BackupEnded;
   248 				break;
   249 			case( EFeatMgrBURState_RestoreStarted ):
   250 				status = EFeatMgrBURState_RestoreEnded;
   251 				break;
   252 			case( EFeatMgrBURState_None ):
   253 				status = EFeatMgrBURState_None;
   254 				break;
   255 			default:
   256 				status = EFeatMgrBURState_Error;
   257 				break;
   258 			}
   259 		}
   260 	else if( (iBURLockFlag & MBackupObserver::EReleaseLockReadOnly) &&
   261 		(iBURLockFlag & ~MBackupObserver::EReleaseLockNoAccess) )
   262 		{
   263 		// starting (making the file read-only is a sign that we are
   264 		// in "backup" mode).
   265 		INFO_LOG("ChangeFileLockL() backup starting");
   266 		status = EFeatMgrBURState_BackupStarted;
   267 		}
   268 	else if( iBURLockFlag & MBackupObserver::EReleaseLockNoAccess )
   269 		{
   270 		// starting (now read/write access is used to signify that a
   271 		// restore has been started).
   272 		INFO_LOG("ChangeFileLockL() restore starting");
   273 		status = EFeatMgrBURState_RestoreStarted;
   274 		}
   275 	else
   276 		{
   277 		// unhandled flag states
   278 		INFO_LOG("ChangeFileLockL() error state");
   279 		status = EFeatMgrBURState_Error;
   280 		}
   281 	
   282 	return status;
   283 	}
   284 
   285 /**
   286  * This function is called from within featmgr server when a backup event occurs
   287  * There are two state variables coming into Feature Manager as types MBackupObserver::TFileLockFlags 
   288  * and TOperationType. Only certain combinations of these two state variables are valid for Feature
   289  * Manager and if the combination is not valid (because there is no way of returning an error) 
   290  * it will just set the internal state machine into an "error" state.
   291  * Given our current state and our goto state (i.e. where we are at the moment and where we want to goto)
   292  * the state machine checks that this is a valid path in our state machine and then perform the correct 
   293  * operations to get us to the next valid state.
   294  */
   295 void CBurState::BUROperationL( const TBackupOperationAttributes& aBackupOperationAttributes )
   296 	{
   297 	BURStatus gotoState    = EFeatMgrBURState_Error;  // fail safe
   298 	BURStatus currentState = iCurrentBURStatus;       // where in the state machine we are at the moment
   299 	
   300 	// Check that the arguments are within valid limits
   301 	if( KErrNone == CheckBURArguments( aBackupOperationAttributes ) )
   302 		{
   303 		// Now convert into states that the Feature Manager understands.
   304 		gotoState = ConvertToBURState( aBackupOperationAttributes );
   305 		}
   306 	
   307 	// Check that this is a valid path and then perform the correct operations to get to the next valid state
   308 	iCurrentBURStatus = BUR_StateMachine( currentState, gotoState );
   309 	
   310 	return;
   311 	}
   312 
   313 /**
   314  * The state machine for the backup and restore of the featmgr 
   315  * The logic that goes into changing states in the state machine.
   316  */
   317 BURStatus CBurState::BUR_StateMachine( BURStatus aCurrent, BURStatus aGoto )
   318 	{
   319 	TInt count = 0;
   320 	TBool found = EFalse;
   321 
   322 	// Fail safe default state values
   323 	BURStatus newState			= EFeatMgrBURState_Error;
   324 	BURStatus stateTableGoto    = EFeatMgrBURState_Error;
   325 	BURStatus stateTableCurrent = EFeatMgrBURState_Error;
   326 	
   327 	// Normal operation
   328 	//
   329 	do
   330 		{
   331 		// Get the n-th state table [current,goto] state vector
   332 		stateTableCurrent = iBURStructArray[count].iCurrent;
   333 		stateTableGoto    = iBURStructArray[count].iGoto;
   334 		
   335 		// if the state table matches what we are given
   336 		if( (aCurrent == stateTableCurrent) && 
   337 			(aGoto    == stateTableGoto) )
   338 			{
   339 			// process the relevant state function
   340 			if( NULL != iBURStructArray[count].iFunc )
   341 				{
   342 				newState = (iServer->*iBURStructArray[count].iFunc)( aCurrent );
   343 
   344 				// check result: when from the state machine is not what we expected from the state table
   345 				if( newState != stateTableGoto )
   346 					{
   347 					// put state machine into an error state and break "normal" loop
   348 					aCurrent = newState;
   349 					aGoto    = EFeatMgrBURState_Error;
   350 					break;
   351 					}
   352 				else
   353 					{
   354 					found = ETrue;
   355 					}
   356 				}
   357 			
   358 			// self-perpetuate in certain cases
   359 			if( (EFeatMgrBURState_BackupEnded == newState) || 
   360 				(EFeatMgrBURState_RestoreEnded == newState) )
   361 				{
   362 				aCurrent = newState;
   363 				aGoto = EFeatMgrBURState_None;
   364 				found = EFalse;
   365 				count = 0;
   366 				}
   367 			}
   368 		count++;
   369 		}
   370 	while( (!found) && count < KBURArrayLength );
   371 	
   372 
   373 	// Error recovery
   374 	//
   375 	if( EFeatMgrBURState_Error == aGoto ||
   376 		EFalse == found )
   377 		{
   378 		// reset 
   379 		aGoto = EFeatMgrBURState_Error;
   380 		count = 0;
   381 		found = EFalse;
   382 
   383 		do
   384 			{
   385 			// Get the n-th state table [current,goto] state vector in the error array
   386 			stateTableCurrent = iBURStructErrorArray[count].iCurrent;
   387 			stateTableGoto    = iBURStructErrorArray[count].iGoto;
   388 			
   389 			// if the state table matches what we are given
   390 			if( ((aCurrent == stateTableCurrent) && (aGoto == stateTableGoto))  )
   391 				{
   392 				// process the relevant state function
   393 				if( NULL != iBURStructErrorArray[count].iFunc )
   394 					{
   395 					newState = (iServer->*iBURStructErrorArray[count].iFunc)( aCurrent );
   396 					// there is no error recovery from error recovery.
   397 					found = ETrue;
   398 					}
   399 				
   400 				// self-perpetuate in certain cases
   401 				if( EFeatMgrBURState_Error == newState )
   402 					{
   403 					aCurrent = newState;
   404 					aGoto = EFeatMgrBURState_None;
   405 					found = EFalse;
   406 					count = 0;
   407 					}
   408 				}
   409 			count++;
   410 			}
   411 		while( (!found) && count < KBURErrorArrayLength );
   412 		
   413 		}
   414 	
   415 	return newState;
   416 	}
   417