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