os/persistentdata/featuremgmt/featuremgr/src/serverexe/featmgrfeatureregistry.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) 2007-2010 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
sl@0
    18
sl@0
    19
// INCLUDE FILES
sl@0
    20
#include <ecom/ecom.h>
sl@0
    21
#include <e32uid.h>
sl@0
    22
#include <f32file.h>
sl@0
    23
#include <e32property.h>
sl@0
    24
#include <sacls.h>
sl@0
    25
#include "featmgrconfiguration.h"
sl@0
    26
#include "featmgrfeatureregistry.h"
sl@0
    27
#include "featmgrserver.h"
sl@0
    28
#include "featmgrdebug.h"
sl@0
    29
sl@0
    30
#define MAXSWIOPS 50
sl@0
    31
#define SWITIMEOUT 15000000
sl@0
    32
sl@0
    33
// CONSTANTS
sl@0
    34
_LIT( KZFeaturesFileNameMatch, "feature*" );
sl@0
    35
_LIT( KCRuntimeFeaturesFileName, "features.dat" );
sl@0
    36
#ifdef EXTENDED_FEATURE_MANAGER_TEST
sl@0
    37
_LIT( KZFeaturesDir, "C:\\Private\\102836E5\\" );
sl@0
    38
_LIT( KCFeatMgrPrivatePath, "?:\\Private\\102836E5\\runtime\\" );
sl@0
    39
#else
sl@0
    40
_LIT( KZFeaturesDir, "Z:\\Private\\10205054\\" );
sl@0
    41
_LIT( KCFeatMgrPrivatePath, "?:\\Private\\10205054\\" );
sl@0
    42
#endif // EXTENDED_FEATURE_MANAGER_TEST
sl@0
    43
sl@0
    44
const TUint32 KDefaultData( 0x00000000 );
sl@0
    45
// Feature file header constants. 
sl@0
    46
// First 4 bytes of config file: ASCII f-e-a-t followed by file version and flags.
sl@0
    47
const TUint32 KFileType( 0x74616566 );
sl@0
    48
const TUint16 KFileVersion( 1 );
sl@0
    49
const TUint16 KFileFlags( 0 );
sl@0
    50
sl@0
    51
sl@0
    52
// ============================= LOCAL FUNCTIONS ===============================
sl@0
    53
sl@0
    54
// ============================ MEMBER FUNCTIONS ===============================
sl@0
    55
sl@0
    56
// -----------------------------------------------------------------------------
sl@0
    57
// CFeatMgrFeatureRegistry::CFeatMgrFeatureRegistry
sl@0
    58
// -----------------------------------------------------------------------------
sl@0
    59
//
sl@0
    60
CFeatMgrFeatureRegistry::CFeatMgrFeatureRegistry( RFs& aFs,
sl@0
    61
    MFeatMgrRegistryObserver& aObserver  )
sl@0
    62
    :
sl@0
    63
    iObserver( aObserver ),
sl@0
    64
    iFs( aFs ),
sl@0
    65
    iSWICacheFeature( EFalse ),
sl@0
    66
    iSWIStatus( ESWIComplete ),
sl@0
    67
    iSWIProcessId( 1 ),
sl@0
    68
    iOomOccured(EFalse)
sl@0
    69
    {
sl@0
    70
    }
sl@0
    71
sl@0
    72
// -----------------------------------------------------------------------------
sl@0
    73
// CFeatMgrFeatureRegistry::ConstructL
sl@0
    74
// Symbian 2nd phase constructor can leave.
sl@0
    75
// -----------------------------------------------------------------------------
sl@0
    76
//
sl@0
    77
void CFeatMgrFeatureRegistry::ConstructL()
sl@0
    78
    {
sl@0
    79
    }
sl@0
    80
sl@0
    81
// -----------------------------------------------------------------------------
sl@0
    82
// CFeatMgrFeatureRegistry::NewL
sl@0
    83
// Two-phased constructor.
sl@0
    84
// -----------------------------------------------------------------------------
sl@0
    85
//
sl@0
    86
CFeatMgrFeatureRegistry* CFeatMgrFeatureRegistry::NewL( RFs& aFs, 
sl@0
    87
    MFeatMgrRegistryObserver& aObserver )
sl@0
    88
    {
sl@0
    89
    CFeatMgrFeatureRegistry* self = 
sl@0
    90
        new( ELeave ) CFeatMgrFeatureRegistry( aFs, aObserver );
sl@0
    91
    
sl@0
    92
    CleanupStack::PushL( self );
sl@0
    93
    self->ConstructL();
sl@0
    94
    CleanupStack::Pop( self );
sl@0
    95
sl@0
    96
    return self;
sl@0
    97
    }
sl@0
    98
sl@0
    99
// -----------------------------------------------------------------------------
sl@0
   100
// Destructor
sl@0
   101
// -----------------------------------------------------------------------------
sl@0
   102
//
sl@0
   103
CFeatMgrFeatureRegistry::~CFeatMgrFeatureRegistry()
sl@0
   104
    {
sl@0
   105
    FUNC_LOG
sl@0
   106
    
sl@0
   107
    iFeatureList.Close();
sl@0
   108
    iRangeList.Close();
sl@0
   109
    iFeatureListBackup.Close();    
sl@0
   110
    iSWICachedOperations.Close();
sl@0
   111
    if( iSWIListener )
sl@0
   112
    	{
sl@0
   113
        delete iSWIListener;    	
sl@0
   114
    	}
sl@0
   115
    if( iSWITimer )
sl@0
   116
    	{
sl@0
   117
    	delete iSWITimer;
sl@0
   118
    	}
sl@0
   119
    }
sl@0
   120
sl@0
   121
sl@0
   122
// -----------------------------------------------------------------------------
sl@0
   123
// CFeatMgrFeatureRegistry::IsFeatureSupported
sl@0
   124
// -----------------------------------------------------------------------------
sl@0
   125
//  
sl@0
   126
TInt CFeatMgrFeatureRegistry::IsFeatureSupported( TFeatureServerEntry& aFeature )
sl@0
   127
    {
sl@0
   128
    TInt err( KErrNotFound );
sl@0
   129
    const TInt index = SearchFeature( aFeature.FeatureUid() );
sl@0
   130
    
sl@0
   131
    if ( index == KErrNotFound )
sl@0
   132
        {
sl@0
   133
        // Check whether feature in supported ranges
sl@0
   134
        TInt count( iRangeList.Count() );
sl@0
   135
        TUid uid( aFeature.FeatureUid() );
sl@0
   136
        for( TInt i = 0; i < count; i++ )
sl@0
   137
            {
sl@0
   138
            if( (uid.iUid >= iRangeList[i].iLowUid.iUid) && 
sl@0
   139
                (uid.iUid <= iRangeList[i].iHighUid.iUid) )
sl@0
   140
                {
sl@0
   141
                TBitFlags32 flags( 0 );
sl@0
   142
                flags.Assign( EFeatureSupported, KFeatureSupported );
sl@0
   143
                TFeatureServerEntry entry( aFeature.FeatureUid(), flags, KDefaultData );
sl@0
   144
                aFeature = entry;
sl@0
   145
                err = KFeatureSupported;
sl@0
   146
                break;
sl@0
   147
                }
sl@0
   148
            }
sl@0
   149
        }
sl@0
   150
    else if( IsFlagSet( index, EFeatureUninitialized ) )
sl@0
   151
        {
sl@0
   152
        // Supported status bit is not taken into account if feature not yet initialized
sl@0
   153
        err = KErrNotReady;
sl@0
   154
        }
sl@0
   155
    else if ( (index < iFeatureList.Count()) && IsFlagSet( index, EFeatureSupported ) )
sl@0
   156
        {
sl@0
   157
        TBitFlags32 flags = iFeatureList[index].FeatureFlags();
sl@0
   158
        flags.Assign( EFeatureSupported, KFeatureSupported );
sl@0
   159
        TUint32 data = iFeatureList[index].FeatureData();
sl@0
   160
        TFeatureServerEntry entry( aFeature.FeatureUid(), flags, data );
sl@0
   161
        aFeature = entry;
sl@0
   162
        err = KFeatureSupported;
sl@0
   163
        }
sl@0
   164
    else
sl@0
   165
        {
sl@0
   166
        TBitFlags32 flags = iFeatureList[index].FeatureFlags();
sl@0
   167
        flags.Assign( EFeatureSupported, KFeatureUnsupported );
sl@0
   168
        TUint32 data = iFeatureList[index].FeatureData();
sl@0
   169
        TFeatureServerEntry entry( aFeature.FeatureUid(), flags, data );
sl@0
   170
        aFeature = entry;
sl@0
   171
        err = KFeatureUnsupported;
sl@0
   172
        }
sl@0
   173
sl@0
   174
    return err;
sl@0
   175
    }
sl@0
   176
sl@0
   177
// -----------------------------------------------------------------------------
sl@0
   178
// CFeatMgrFeatureRegistry::AddFeature
sl@0
   179
// -----------------------------------------------------------------------------
sl@0
   180
//
sl@0
   181
TInt CFeatMgrFeatureRegistry::AddFeature( TFeatureServerEntry& aFeature, TUint aPrcId )
sl@0
   182
    {
sl@0
   183
    TInt err( KErrAccessDenied );
sl@0
   184
    
sl@0
   185
    if( iSWIProcessId == aPrcId && iSWICacheFeature )
sl@0
   186
		{
sl@0
   187
		err = SWICacheCommand(ESWIAddFeat, aFeature);
sl@0
   188
		}
sl@0
   189
    else
sl@0
   190
    	{
sl@0
   191
        const TInt index = SearchFeature( aFeature.FeatureUid() );
sl@0
   192
	
sl@0
   193
	    if ( index == KErrNotFound )
sl@0
   194
	        {
sl@0
   195
	        TBitFlags32 flags( aFeature.FeatureFlags() );
sl@0
   196
	        flags.Set( EFeatureRuntime );
sl@0
   197
	
sl@0
   198
	        //Check the feature falg is valid
sl@0
   199
	        TRAP(err,ValidateRuntimeFeatureFlagL(flags));
sl@0
   200
	        if (err != KErrNone)
sl@0
   201
	            return err;
sl@0
   202
	
sl@0
   203
	        TFeatureServerEntry entry( aFeature.FeatureUid(), flags, aFeature.FeatureData() );	       
sl@0
   204
			err = iFeatureList.InsertInOrder( entry, FindByUid );
sl@0
   205
	        if ( err == KErrNone )
sl@0
   206
	            {
sl@0
   207
	            TFeatureChangeType changeType( EFeatureFeatureCreated );
sl@0
   208
	            err = HandleChange( entry, changeType );
sl@0
   209
	            }
sl@0
   210
	        }
sl@0
   211
	    else
sl@0
   212
	        {
sl@0
   213
	        err = KErrAlreadyExists;
sl@0
   214
	        }
sl@0
   215
	    
sl@0
   216
	    INFO_LOG("CFeatMgrFeatureRegistry::AddFeature - Features directly stored in registry");
sl@0
   217
    	}
sl@0
   218
    
sl@0
   219
    LOG_IF_ERROR1( err, "CFeatMgrFeatureRegistry::AddFeature - result %d", err );
sl@0
   220
sl@0
   221
    return err;
sl@0
   222
    }
sl@0
   223
sl@0
   224
// -----------------------------------------------------------------------------
sl@0
   225
// CFeatMgrFeatureRegistry::DeleteFeature
sl@0
   226
// -----------------------------------------------------------------------------
sl@0
   227
//
sl@0
   228
TInt CFeatMgrFeatureRegistry::DeleteFeature( TUid aFeature, TUint aPrcId  )
sl@0
   229
    {
sl@0
   230
    TInt err( KErrAccessDenied );
sl@0
   231
	
sl@0
   232
    if( iSWIProcessId == aPrcId && iSWICacheFeature )
sl@0
   233
    	{
sl@0
   234
    	err = SWICacheCommand(ESWIDeleteFeat, aFeature);
sl@0
   235
    	}
sl@0
   236
    else
sl@0
   237
    	{
sl@0
   238
	    // Check if the feature is runtime
sl@0
   239
	    TInt index = SearchFeature( aFeature );
sl@0
   240
	    if ( index == KErrNotFound )
sl@0
   241
	        {
sl@0
   242
	        return KErrNotFound;
sl@0
   243
	        }
sl@0
   244
	    if ( !iFeatureList[index].FeatureFlags().IsSet(EFeatureRuntime) )
sl@0
   245
	        {
sl@0
   246
	        return KErrAccessDenied;
sl@0
   247
	        }
sl@0
   248
	
sl@0
   249
	    TFeatureServerEntry entry = iFeatureList[index];
sl@0
   250
	    
sl@0
   251
    	iFeatureList.Remove( index );
sl@0
   252
        TFeatureChangeType changeType( EFeatureFeatureDeleted );
sl@0
   253
        err = HandleChange( entry, changeType );
sl@0
   254
        
sl@0
   255
        INFO_LOG("CFeatMgrFeatureRegistry::DeleteFeature - Features deleted directly from registry");
sl@0
   256
        }
sl@0
   257
    
sl@0
   258
    LOG_IF_ERROR1( err, "CFeatMgrFeatureRegistry::DeleteFeature - result %d", err );
sl@0
   259
    
sl@0
   260
    return err;
sl@0
   261
    }
sl@0
   262
sl@0
   263
// -----------------------------------------------------------------------------
sl@0
   264
// CFeatMgrFeatureRegistry::SetFeature
sl@0
   265
//This method cannot set feature flag range in DSR unless it is added to before
sl@0
   266
//using CFeatMgrFeatureRegistry::AddFeature()
sl@0
   267
// -----------------------------------------------------------------------------
sl@0
   268
//	
sl@0
   269
TInt CFeatMgrFeatureRegistry::SetFeature( TUid aFeature, TInt aEnable, const TUint32 *aData, TUint aPrcId  )
sl@0
   270
    {
sl@0
   271
    FUNC_LOG
sl@0
   272
sl@0
   273
    TInt err( KErrNone );
sl@0
   274
    
sl@0
   275
    if( iSWIProcessId == aPrcId && iSWICacheFeature )
sl@0
   276
    	{
sl@0
   277
        TBitFlags32 flags(0);
sl@0
   278
        flags.Assign( EFeatureSupported, aEnable );
sl@0
   279
        TUint32 data = 0;
sl@0
   280
        if( aData )
sl@0
   281
            {
sl@0
   282
            data = *aData;
sl@0
   283
            }
sl@0
   284
sl@0
   285
        TFeatureServerEntry entry( aFeature, flags, data );
sl@0
   286
sl@0
   287
        // If aData is not null, we want to change the user data too
sl@0
   288
        //  otherwise only change the feature status.
sl@0
   289
        if( aData )
sl@0
   290
            {
sl@0
   291
            err = SWICacheCommand(ESWISetFeatAndData, entry);
sl@0
   292
            }
sl@0
   293
        else
sl@0
   294
            {
sl@0
   295
            err = SWICacheCommand(ESWISetFeat, entry);
sl@0
   296
            }
sl@0
   297
    	}
sl@0
   298
    else 
sl@0
   299
    	{
sl@0
   300
	    TInt index;
sl@0
   301
	
sl@0
   302
	    // Validate feature exists and is modifiable
sl@0
   303
	    err = ValidateFeature( aFeature, index );
sl@0
   304
	
sl@0
   305
	    if ( err != KErrNone )
sl@0
   306
	        {
sl@0
   307
			return err;
sl@0
   308
	        }
sl@0
   309
	
sl@0
   310
	    if ( (index >= 0 && index < iFeatureList.Count()) )
sl@0
   311
	        {
sl@0
   312
        	TBitFlags32 flags = iFeatureList[index].FeatureFlags();
sl@0
   313
	        TUint32 data = iFeatureList[index].FeatureData();
sl@0
   314
	        TFeatureChangeType changeType( EFeatureStatusUpdated );
sl@0
   315
sl@0
   316
	        // Update "supported" info according to request
sl@0
   317
	        if( (aEnable == EFeatureSupportEnable) || (aEnable == EFeatureSupportDisable) )
sl@0
   318
	            {
sl@0
   319
	            INFO_LOG1( "CFeatMgrFeatureRegistry::SetFeature() - aEnable %d", aEnable );
sl@0
   320
	            flags.Assign( EFeatureSupported, aEnable );
sl@0
   321
	            }
sl@0
   322
	        // When setting feature, always unset "uninitialized" bit
sl@0
   323
	        flags.Assign( EFeatureUninitialized, 0 );
sl@0
   324
sl@0
   325
	        // Update data whenever applied
sl@0
   326
	        if( aData )
sl@0
   327
	            {
sl@0
   328
	            INFO_LOG1( "CFeatMgrFeatureRegistry::SetFeature() - aData %d", aData );
sl@0
   329
	            data = *aData;
sl@0
   330
sl@0
   331
	            if( aEnable == EFeatureSupportUntouch )
sl@0
   332
	                {
sl@0
   333
	                changeType = EFeatureDataUpdated;
sl@0
   334
	                }
sl@0
   335
	            else
sl@0
   336
	                {
sl@0
   337
	                changeType = EFeatureStatusDataUpdated;
sl@0
   338
	                }
sl@0
   339
	            }
sl@0
   340
sl@0
   341
	        TFeatureServerEntry entry( aFeature, flags, data );
sl@0
   342
	        //Check if the feature will actually change
sl@0
   343
	        if(iFeatureList[index].FeatureFlags() == flags && iFeatureList[index].FeatureData() == data )
sl@0
   344
	        	{
sl@0
   345
	        	//No change were made, set change type to EFeatureNoChange
sl@0
   346
	        	changeType = EFeatureNoChange;
sl@0
   347
	        	}
sl@0
   348
	        else
sl@0
   349
	        	{
sl@0
   350
	        	// Set the feature entry in list with updated information
sl@0
   351
	        	iFeatureList[index].Set(entry);
sl@0
   352
	        	}
sl@0
   353
sl@0
   354
	        err = HandleChange( entry, changeType );
sl@0
   355
	        }
sl@0
   356
    	}
sl@0
   357
    
sl@0
   358
    LOG_IF_ERROR1( err, "CFeatMgrFeatureRegistry::SetFeature - result %d", err );
sl@0
   359
sl@0
   360
    return err;
sl@0
   361
    }
sl@0
   362
sl@0
   363
// -----------------------------------------------------------------------------
sl@0
   364
// CFeatMgrFeatureRegistry::HandleChange
sl@0
   365
// -----------------------------------------------------------------------------
sl@0
   366
//  
sl@0
   367
TInt CFeatMgrFeatureRegistry::HandleChange( TFeatureServerEntry& aFeature, 
sl@0
   368
    TFeatureChangeType aType )
sl@0
   369
    {
sl@0
   370
    FUNC_LOG
sl@0
   371
    
sl@0
   372
    TInt err( KErrNone );
sl@0
   373
    
sl@0
   374
    // Update feature file, when feature is specified as persisted.
sl@0
   375
    if ( aFeature.FeatureFlags().IsSet( EFeaturePersisted ) )
sl@0
   376
        {
sl@0
   377
        TRAP( err, UpdateRuntimeFeaturesFileL( aFeature, aType ) );
sl@0
   378
        LOG_IF_ERROR1( err, "CFeatMgrFeatureRegistry::HandleChange - update error %d", err );
sl@0
   379
    
sl@0
   380
        // It is questionnable whether we should remove the feature from iFeatureList.
sl@0
   381
        // However, feature is usable until device is powered down and features reloaded.
sl@0
   382
        // if ( err == KErrNone )
sl@0
   383
        }
sl@0
   384
    
sl@0
   385
    // It is also questionnable whether we should suppress notification in case file 
sl@0
   386
    // update failed.
sl@0
   387
    // if ( err == KErrNone )
sl@0
   388
    iObserver.HandleFeatureChange( aFeature, aType );
sl@0
   389
    
sl@0
   390
    return err;
sl@0
   391
    }
sl@0
   392
    
sl@0
   393
// -----------------------------------------------------------------------------
sl@0
   394
// CFeatMgrFeatureRegistry::ValidateFeature
sl@0
   395
// -----------------------------------------------------------------------------
sl@0
   396
//  
sl@0
   397
TInt CFeatMgrFeatureRegistry::ValidateFeature( TUid aFeature, TInt &aIndex )
sl@0
   398
    {
sl@0
   399
    TInt err( KErrNone );
sl@0
   400
    
sl@0
   401
    aIndex = SearchFeature( aFeature );
sl@0
   402
     
sl@0
   403
    if ( aIndex == KErrNotFound )
sl@0
   404
        {
sl@0
   405
        err = KErrNotFound;
sl@0
   406
        }
sl@0
   407
    else if ( !IsFlagSet( aIndex, EFeatureModifiable ) )
sl@0
   408
        {
sl@0
   409
        err = KErrAccessDenied;
sl@0
   410
        }
sl@0
   411
    
sl@0
   412
    return err;
sl@0
   413
    }
sl@0
   414
    
sl@0
   415
// -----------------------------------------------------------------------------
sl@0
   416
// CFeatMgrFeatureRegistry::FindByUid
sl@0
   417
// Returns Zero if UIDs do match.
sl@0
   418
// -----------------------------------------------------------------------------
sl@0
   419
//  
sl@0
   420
TInt CFeatMgrFeatureRegistry::FindByUid( const TUid *aFeature, 
sl@0
   421
    const TFeatureServerEntry& aItem )
sl@0
   422
    {
sl@0
   423
    if ( aFeature->iUid < aItem.FeatureUid().iUid )
sl@0
   424
        {
sl@0
   425
        return -1;
sl@0
   426
        }
sl@0
   427
    else if ( aFeature->iUid > aItem.FeatureUid().iUid )
sl@0
   428
        {
sl@0
   429
        return 1;
sl@0
   430
        }
sl@0
   431
sl@0
   432
    return 0;
sl@0
   433
    }
sl@0
   434
sl@0
   435
// -----------------------------------------------------------------------------
sl@0
   436
// CFeatMgrFeatureRegistry::FindByUid
sl@0
   437
// Returns Zero if UIDs do match.
sl@0
   438
// -----------------------------------------------------------------------------
sl@0
   439
//  
sl@0
   440
TInt CFeatMgrFeatureRegistry::FindByUid( const TFeatureServerEntry& aFeature, 
sl@0
   441
    const TFeatureServerEntry& aItem )
sl@0
   442
    {
sl@0
   443
    if ( aFeature.FeatureUid().iUid < aItem.FeatureUid().iUid )
sl@0
   444
        {
sl@0
   445
        return -1;
sl@0
   446
        }
sl@0
   447
    else if ( aFeature.FeatureUid().iUid > aItem.FeatureUid().iUid )
sl@0
   448
        {
sl@0
   449
        return 1;
sl@0
   450
        }
sl@0
   451
sl@0
   452
    return 0;
sl@0
   453
    }
sl@0
   454
sl@0
   455
// -----------------------------------------------------------------------------
sl@0
   456
// CFeatMgrFeatureRegistry::SearchFeature
sl@0
   457
// -----------------------------------------------------------------------------
sl@0
   458
//  
sl@0
   459
TInt CFeatMgrFeatureRegistry::SearchFeature( TUid aFeature )
sl@0
   460
    {
sl@0
   461
    const TUid& uid( aFeature );
sl@0
   462
    return iFeatureList.FindInOrder( uid, FindByUid );
sl@0
   463
    }
sl@0
   464
    
sl@0
   465
// -----------------------------------------------------------------------------
sl@0
   466
// CFeatMgrFeatureRegistry::IsFlagSet
sl@0
   467
// -----------------------------------------------------------------------------
sl@0
   468
//  
sl@0
   469
TBool CFeatMgrFeatureRegistry::IsFlagSet( TInt aIndex, TFeatureFlags aFlag )
sl@0
   470
    {
sl@0
   471
    TBool isSet( EFalse );
sl@0
   472
    if( aIndex < iFeatureList.Count() )
sl@0
   473
        {
sl@0
   474
        isSet = iFeatureList[aIndex].FeatureFlags().IsSet(aFlag);
sl@0
   475
        }
sl@0
   476
        
sl@0
   477
    return isSet;
sl@0
   478
    }
sl@0
   479
sl@0
   480
// -----------------------------------------------------------------------------
sl@0
   481
// CFeatMgrFeatureRegistry::SupportedFeatures
sl@0
   482
// -----------------------------------------------------------------------------
sl@0
   483
//  
sl@0
   484
void CFeatMgrFeatureRegistry::SupportedFeaturesL( RFeatureUidArray& aSupportedFeatures )
sl@0
   485
    {
sl@0
   486
    FUNC_LOG
sl@0
   487
    
sl@0
   488
    TInt count = iFeatureList.Count();
sl@0
   489
    
sl@0
   490
    for ( TInt i = 0; i < count; i++ )
sl@0
   491
        {
sl@0
   492
        if( IsFlagSet( i, EFeatureSupported) )
sl@0
   493
            {
sl@0
   494
            aSupportedFeatures.AppendL( iFeatureList[i].FeatureUid() );
sl@0
   495
            }
sl@0
   496
        }
sl@0
   497
    }
sl@0
   498
sl@0
   499
// -----------------------------------------------------------------------------
sl@0
   500
// CFeatMgrFeatureRegistry::NumberOfSupportedFeatures
sl@0
   501
// -----------------------------------------------------------------------------
sl@0
   502
//  
sl@0
   503
TInt CFeatMgrFeatureRegistry::NumberOfSupportedFeatures()
sl@0
   504
    {
sl@0
   505
    FUNC_LOG
sl@0
   506
    
sl@0
   507
    TInt count = iFeatureList.Count();
sl@0
   508
    TInt countSupported(0);
sl@0
   509
    
sl@0
   510
    for ( TInt i = 0; i < count; i++ )
sl@0
   511
        {
sl@0
   512
        if( IsFlagSet( i, EFeatureSupported) )
sl@0
   513
            {
sl@0
   514
            countSupported++;
sl@0
   515
            }
sl@0
   516
        }
sl@0
   517
    
sl@0
   518
    return countSupported;
sl@0
   519
    }
sl@0
   520
sl@0
   521
// -----------------------------------------------------------------------------
sl@0
   522
sl@0
   523
void CFeatMgrFeatureRegistry::ResetFeaturesL()
sl@0
   524
    {
sl@0
   525
    FUNC_LOG
sl@0
   526
    
sl@0
   527
    // backup the feature list before it is destroyed
sl@0
   528
    iFeatureListBackup.Reset();
sl@0
   529
    const TInt KCount = iFeatureList.Count();
sl@0
   530
   	iFeatureListBackup.ReserveL(KCount);
sl@0
   531
	for(TInt i(0); i < KCount; i++)
sl@0
   532
		{
sl@0
   533
		// The main error here would be KErrNoMemory which should not happen as
sl@0
   534
		// we have already reserved the space. However, we should still check.
sl@0
   535
		iFeatureListBackup.AppendL(iFeatureList[i]);
sl@0
   536
		}
sl@0
   537
	    
sl@0
   538
    // destroy the feature list
sl@0
   539
    iFeatureList.Reset();
sl@0
   540
    iFeatureList.Close();
sl@0
   541
    
sl@0
   542
    iRangeList.Reset();
sl@0
   543
    iRangeList.Close();
sl@0
   544
    }
sl@0
   545
sl@0
   546
/**
sl@0
   547
 * Get the fully qualified path and filename to the features.dat
sl@0
   548
 * data file.
sl@0
   549
 */ 
sl@0
   550
TFileName CFeatMgrFeatureRegistry::GetFeaturesFilePathAndName( void )
sl@0
   551
	{
sl@0
   552
    TFileName path( KCFeatMgrPrivatePath );
sl@0
   553
sl@0
   554
    path[0] = iFs.GetSystemDriveChar();
sl@0
   555
    path.Append( KCRuntimeFeaturesFileName );
sl@0
   556
sl@0
   557
    return path;
sl@0
   558
	}
sl@0
   559
sl@0
   560
// CFeatMgrFeatureRegistry::ReadFeatureFilesL
sl@0
   561
// Reads platform and product feature files. 
sl@0
   562
// -----------------------------------------------------------------------------
sl@0
   563
//  
sl@0
   564
void CFeatMgrFeatureRegistry::ReadFeatureFilesL()
sl@0
   565
    {
sl@0
   566
    FUNC_LOG    
sl@0
   567
    
sl@0
   568
    // Read feature files from Z
sl@0
   569
    ReadFilesFromDirL( KZFeaturesDir );
sl@0
   570
sl@0
   571
    //check that there is at least one DSR 
sl@0
   572
    if (!iRangeList.Count()) 
sl@0
   573
    	{
sl@0
   574
    	_LIT(KPanicCategory, "FeatMgrServer");
sl@0
   575
    	ERROR_LOG( "CFeatMgrFeatureRegistry::ReadFilesFromDirL() - no DSR found in ROM; this indicates a system integration error  - going to panic" );
sl@0
   576
    	::FmgrFatalErrorL(KErrNotFound, KPanicCategory, EPanicNoDSR);    
sl@0
   577
    	}
sl@0
   578
    }
sl@0
   579
sl@0
   580
// -----------------------------------------------------------------------------
sl@0
   581
// CFeatMgrFeatureRegistry::ReadFilesFromDirL
sl@0
   582
// -----------------------------------------------------------------------------
sl@0
   583
//
sl@0
   584
void CFeatMgrFeatureRegistry::ReadFilesFromDirL( const TDesC& aDirName )
sl@0
   585
    {
sl@0
   586
    _LIT(KPanicCategory, "FEATMGR-READFILE");
sl@0
   587
sl@0
   588
    CDir* dir = NULL; 
sl@0
   589
    TInt err( KErrNone );
sl@0
   590
sl@0
   591
    err = iFs.GetDir( aDirName, KEntryAttNormal, ESortByName, dir );
sl@0
   592
    CleanupStack::PushL( dir );
sl@0
   593
sl@0
   594
    if( err == KErrNone )
sl@0
   595
        {
sl@0
   596
        err = ReadFiles( aDirName, dir );
sl@0
   597
        if ( err != KErrNone )
sl@0
   598
            {            
sl@0
   599
            ERROR_LOG1( "CFeatMgrFeatureRegistry::ReadFilesFromDirL() - err %d ", err );
sl@0
   600
            User::Leave( err );    
sl@0
   601
            }
sl@0
   602
        }  
sl@0
   603
    else if( err == KErrPathNotFound )
sl@0
   604
    	{
sl@0
   605
    	::FmgrFatalErrorL(err, KPanicCategory, EPanicNoFeatureFiles);
sl@0
   606
    	}
sl@0
   607
	else
sl@0
   608
        {            
sl@0
   609
        ERROR_LOG1( "CFeatMgrFeatureRegistry::ReadFilesFromDirL() - err %d ", err );
sl@0
   610
        User::Leave( err );    
sl@0
   611
        }
sl@0
   612
    
sl@0
   613
    CleanupStack::PopAndDestroy( dir );
sl@0
   614
    }
sl@0
   615
sl@0
   616
// -----------------------------------------------------------------------------
sl@0
   617
// CFeatMgrFeatureRegistry::ReadRuntimeFeaturesL
sl@0
   618
// -----------------------------------------------------------------------------
sl@0
   619
//
sl@0
   620
void CFeatMgrFeatureRegistry::ReadRuntimeFeaturesL( TBool &aFeaturesReady )
sl@0
   621
    {
sl@0
   622
    TFileName path( KCFeatMgrPrivatePath );
sl@0
   623
    path[0] = iFs.GetSystemDriveChar();
sl@0
   624
    path.Append( KCRuntimeFeaturesFileName );
sl@0
   625
sl@0
   626
    TInt err( KErrNone );
sl@0
   627
    TRAP( err, ReadFileL( path ) );
sl@0
   628
sl@0
   629
    if ((err == KErrCorrupt) || (err == KErrArgument))
sl@0
   630
    	{
sl@0
   631
    	User::LeaveIfError(iFs.Delete(path));
sl@0
   632
    	aFeaturesReady = ETrue;
sl@0
   633
    	}
sl@0
   634
    else if ( err != KErrNone && err != KErrNotFound && err != KErrPathNotFound )
sl@0
   635
        {  
sl@0
   636
        ERROR_LOG1( "CFeatMgrFeatureRegistry::ReadRuntimeFeatures - ReadFileL returned err %d", err );
sl@0
   637
        User::Leave( err );
sl@0
   638
        }
sl@0
   639
    else
sl@0
   640
        {
sl@0
   641
        aFeaturesReady = ETrue;
sl@0
   642
        }
sl@0
   643
    }
sl@0
   644
sl@0
   645
// -----------------------------------------------------------------------------
sl@0
   646
// CFeatMgrFeatureRegistry::ReadFiles
sl@0
   647
// -----------------------------------------------------------------------------
sl@0
   648
//  
sl@0
   649
TInt CFeatMgrFeatureRegistry::ReadFiles( const TDesC& aPath, CDir* aDir )
sl@0
   650
    {
sl@0
   651
    TInt fileCount = aDir->Count();
sl@0
   652
    TFileName fileName;
sl@0
   653
    TInt err( KErrNotFound );
sl@0
   654
    
sl@0
   655
    for ( TInt file = 0; file < fileCount; file++ )
sl@0
   656
        {
sl@0
   657
        TInt match = (*aDir)[file].iName.MatchC( KZFeaturesFileNameMatch );
sl@0
   658
        if( match != KErrNotFound )
sl@0
   659
            {
sl@0
   660
            fileName.Copy(aPath);
sl@0
   661
            fileName.Append((*aDir)[file].iName);
sl@0
   662
sl@0
   663
            INFO_LOG1( "CFeatMgrFeatureRegistry::ReadFiles - file: %S", &fileName );
sl@0
   664
            TRAP( err, ReadFileL( fileName ) );
sl@0
   665
            LOG_IF_ERROR2( err, "CFeatMgrFeatureRegistry::ReadFiles - file: %S, err %d", 
sl@0
   666
                &fileName, err );
sl@0
   667
            
sl@0
   668
            // Return error if reading of any feature file fails.
sl@0
   669
            if( err != KErrNone )
sl@0
   670
                {
sl@0
   671
                break;
sl@0
   672
                }
sl@0
   673
            }
sl@0
   674
        }
sl@0
   675
sl@0
   676
    return( err );
sl@0
   677
    }
sl@0
   678
sl@0
   679
// -----------------------------------------------------------------------------
sl@0
   680
// CFeatMgrFeatureRegistry::ReadFileL
sl@0
   681
// -----------------------------------------------------------------------------
sl@0
   682
//  
sl@0
   683
void CFeatMgrFeatureRegistry::ReadFileL( const TDesC& aFullPath )
sl@0
   684
    {
sl@0
   685
    FUNC_LOG
sl@0
   686
   
sl@0
   687
    TUint32 count;
sl@0
   688
 	RFileReadStream readStream;
sl@0
   689
sl@0
   690
 	// Open the file and attach to stream 
sl@0
   691
    User::LeaveIfError( readStream.Open( iFs, aFullPath, EFileRead ) );
sl@0
   692
    CleanupClosePushL( readStream );
sl@0
   693
    TUint32 countDSRs;
sl@0
   694
    
sl@0
   695
    //Validate the header
sl@0
   696
    ValidateHeaderL( readStream, count, countDSRs );
sl@0
   697
sl@0
   698
    RArray<TFeatureServerEntry> tempFeatureArray;
sl@0
   699
    CleanupClosePushL( tempFeatureArray );
sl@0
   700
    
sl@0
   701
    //Find the directory that the feature file is contained in. 
sl@0
   702
    TFileName dirName(aFullPath);
sl@0
   703
    TChar toFind = '\\';
sl@0
   704
    dirName.Delete((dirName.LocateReverse(toFind)+1), dirName.Length() );
sl@0
   705
    TBool runtimeFile = EFalse;
sl@0
   706
    if (dirName.Compare(KZFeaturesDir) != 0) //Location of the feature file.
sl@0
   707
    	{
sl@0
   708
    	runtimeFile = ETrue;
sl@0
   709
    	}
sl@0
   710
    
sl@0
   711
    tempFeatureArray.ReserveL(count);
sl@0
   712
    
sl@0
   713
	for(TInt i = 0; i < count; i++)
sl@0
   714
        {
sl@0
   715
        TFeatureServerEntry entry;
sl@0
   716
        entry.InternalizeL( readStream );
sl@0
   717
        
sl@0
   718
        //Check for feature flag errors
sl@0
   719
        TBitFlags32 flags =  entry.FeatureFlags();
sl@0
   720
        TInt err = KErrNone;
sl@0
   721
        
sl@0
   722
        //Validate the flags
sl@0
   723
        // This validation is done in this read function because the validation functions used
sl@0
   724
        // are called in other places were this validation is not appropriate. 
sl@0
   725
        if (runtimeFile)
sl@0
   726
        	{
sl@0
   727
        	if (!flags.IsSet(EFeatureRuntime)) //Check to see if the Runtime flag is set if it is not then the feature should have been read in from the rom. 
sl@0
   728
        		{
sl@0
   729
        		if (SearchFeature( entry.FeatureUid() ) == KErrNotFound )// Check to see if the feature has been read in previously from the rom.
sl@0
   730
        			{
sl@0
   731
        			User::Leave(KErrCorrupt); 
sl@0
   732
        			}
sl@0
   733
        		else //The feature has not been read in previously from the rom file and is therefore invalid. The file is deemed to be corrupt
sl@0
   734
        			{
sl@0
   735
        			ValidateRuntimeFeatureFlagL(flags);   
sl@0
   736
            		}
sl@0
   737
        		}
sl@0
   738
        	else //Flag set the feature is runtime this is then validated as normal
sl@0
   739
        		{
sl@0
   740
        		ValidateRuntimeFeatureFlagL(flags);    	
sl@0
   741
        		}
sl@0
   742
sl@0
   743
        	}
sl@0
   744
        else //File is not as runtime file.
sl@0
   745
        	{
sl@0
   746
        	ValidateFeatureFlagL(flags);
sl@0
   747
        	}
sl@0
   748
        
sl@0
   749
        //If a feature flag defined in system drive (c:) is invalid, it will not be added to Feature Manager 
sl@0
   750
        if ( (err != KErrNone) && flags.IsSet(EFeatureRuntime) )
sl@0
   751
        	{
sl@0
   752
        	continue;
sl@0
   753
       		}
sl@0
   754
        
sl@0
   755
		tempFeatureArray.InsertL( entry, i);
sl@0
   756
        }
sl@0
   757
sl@0
   758
    // Reserve memory if list still empty
sl@0
   759
    if( !iFeatureList.Count() )
sl@0
   760
        {
sl@0
   761
        iFeatureList.ReserveL( tempFeatureArray.Count() );
sl@0
   762
        }
sl@0
   763
sl@0
   764
    // Read features from temp array
sl@0
   765
    for(TInt i = 0; i < tempFeatureArray.Count(); i++)
sl@0
   766
        {
sl@0
   767
        TFeatureServerEntry entry = tempFeatureArray[i];
sl@0
   768
 
sl@0
   769
        TInt index = SearchFeature( entry.FeatureUid() );
sl@0
   770
        
sl@0
   771
        if( index == KErrNotFound)
sl@0
   772
            {
sl@0
   773
            iFeatureList.InsertInOrderL( entry, FindByUid );
sl@0
   774
            }
sl@0
   775
        else
sl@0
   776
            {
sl@0
   777
            INFO_LOG1( "CFeatMgrFeatureRegistry::ReadFileL - replacing uid 0x%08x",
sl@0
   778
                iFeatureList[index].FeatureUid().iUid );
sl@0
   779
            // Set the feature if it is not previously blacklisted
sl@0
   780
            if ( !IsFlagSet( index, EFeatureBlackListed ) )
sl@0
   781
                {
sl@0
   782
                iFeatureList[index].Set(entry);
sl@0
   783
                }
sl@0
   784
            }
sl@0
   785
        }
sl@0
   786
    
sl@0
   787
    // Reserve memory if DSR list still empty
sl@0
   788
    if( !iRangeList.Count() )
sl@0
   789
        {
sl@0
   790
        iRangeList.ReserveL( countDSRs );
sl@0
   791
        }
sl@0
   792
    
sl@0
   793
    // Read default supported ranges from file
sl@0
   794
    for(TInt i = 0; i < countDSRs; i++)
sl@0
   795
        {
sl@0
   796
        TDefaultRange range;
sl@0
   797
        range.iLowUid = TUid::Uid( readStream.ReadUint32L() );
sl@0
   798
        range.iHighUid = TUid::Uid( readStream.ReadUint32L() );
sl@0
   799
        iRangeList.AppendL( range );
sl@0
   800
        if( iRangeList[i].iLowUid.iUid > iRangeList[i].iHighUid.iUid )
sl@0
   801
            {
sl@0
   802
            ERROR_LOG( "CFeatMgrFeatureRegistry::ReadFileL - invalid supported range" );
sl@0
   803
            iRangeList.Remove( i );
sl@0
   804
            User::Leave( KErrCorrupt );
sl@0
   805
            }
sl@0
   806
        }
sl@0
   807
sl@0
   808
#if defined(FEATMGR_INFO_LOG_ENABLED)
sl@0
   809
    count = iFeatureList.Count();
sl@0
   810
    INFO_LOG1( "CFeatMgrFeatureRegistry::ReadFileL - feature entries: %d", count );
sl@0
   811
    for(TInt i = 0; i < count; i++)
sl@0
   812
        {
sl@0
   813
        INFO_LOG3( "CFeatMgrFeatureRegistry::ReadFileL - uid 0x%08x, flags %d, data %d",
sl@0
   814
            iFeatureList[i].FeatureUid().iUid, iFeatureList[i].FeatureFlags().iFlags, 
sl@0
   815
            iFeatureList[i].FeatureData() );
sl@0
   816
        }
sl@0
   817
sl@0
   818
    count = iRangeList.Count();
sl@0
   819
    INFO_LOG1( "CFeatMgrFeatureRegistry::ReadFileL - supported ranges: %d", count );
sl@0
   820
    for(TInt i = 0; i < count; i++)
sl@0
   821
        {
sl@0
   822
        INFO_LOG2( "CFeatMgrFeatureRegistry::ReadFileL - low 0x%08x, high 0x%08x",
sl@0
   823
            iRangeList[i].iLowUid, iRangeList[i].iHighUid );
sl@0
   824
        }
sl@0
   825
#endif
sl@0
   826
sl@0
   827
    CleanupStack::PopAndDestroy( &tempFeatureArray);
sl@0
   828
    CleanupStack::PopAndDestroy( &readStream );
sl@0
   829
sl@0
   830
    }
sl@0
   831
sl@0
   832
// -----------------------------------------------------------------------------
sl@0
   833
// CFeatMgrFeatureRegistry::ValidateHeaderL
sl@0
   834
// -----------------------------------------------------------------------------
sl@0
   835
//  
sl@0
   836
void CFeatMgrFeatureRegistry::ValidateHeaderL( RFileReadStream &aStream, 
sl@0
   837
    TUint32& aCount, TUint32& aCountDSRs )
sl@0
   838
    {
sl@0
   839
    FUNC_LOG
sl@0
   840
sl@0
   841
    TUint32 identifier = aStream.ReadUint32L();
sl@0
   842
    TUint16 fileVersion = aStream.ReadUint16L();
sl@0
   843
    TUint16 fileFlags = aStream.ReadUint16L();
sl@0
   844
    aCount = aStream.ReadUint32L();
sl@0
   845
    aCountDSRs = aStream.ReadUint32L();
sl@0
   846
        
sl@0
   847
    // Carry out simple verification of file content
sl@0
   848
    if((identifier != KFileType) || fileVersion != KFileVersion || 
sl@0
   849
        fileFlags != KFileFlags )
sl@0
   850
        {
sl@0
   851
        User::Leave( KErrCorrupt );
sl@0
   852
        }
sl@0
   853
    }
sl@0
   854
sl@0
   855
// -----------------------------------------------------------------------------
sl@0
   856
// CFeatMgrFeatureRegistry::UpdateRuntimeFeaturesFileL
sl@0
   857
// -----------------------------------------------------------------------------
sl@0
   858
//  
sl@0
   859
void CFeatMgrFeatureRegistry::UpdateRuntimeFeaturesFileL( TFeatureServerEntry& aFeature,
sl@0
   860
		TFeatureChangeType aType )
sl@0
   861
    {
sl@0
   862
    FUNC_LOG
sl@0
   863
sl@0
   864
    // Opens a file containing a stream and prepares the stream for writing.
sl@0
   865
    TInt err( KErrNone );
sl@0
   866
    RFileWriteStream writeStream;
sl@0
   867
    TFileName path( KCFeatMgrPrivatePath );
sl@0
   868
    path[0] = iFs.GetSystemDriveChar();
sl@0
   869
    path.Append( KCRuntimeFeaturesFileName );
sl@0
   870
sl@0
   871
    err = writeStream.Open( iFs, path, EFileWrite  ); 
sl@0
   872
    CleanupClosePushL( writeStream );
sl@0
   873
    
sl@0
   874
    if( err == KErrPathNotFound || err == KErrNotFound )
sl@0
   875
    	{
sl@0
   876
        // Create folder and file.
sl@0
   877
    	if ( err == KErrPathNotFound )
sl@0
   878
    		{
sl@0
   879
    		path = KCFeatMgrPrivatePath;
sl@0
   880
    		path[0] = iFs.GetSystemDriveChar();
sl@0
   881
		    User::LeaveIfError( iFs.MkDirAll( path ) );
sl@0
   882
            path.Append( KCRuntimeFeaturesFileName );
sl@0
   883
    		}
sl@0
   884
        User::LeaveIfError( writeStream.Create( iFs, path, EFileWrite ) );
sl@0
   885
        
sl@0
   886
        // Write header and entry
sl@0
   887
        RFeatureServerArray temp(1);
sl@0
   888
        CleanupClosePushL( temp );
sl@0
   889
        temp.AppendL( aFeature );
sl@0
   890
        WriteHeaderAndEntriesL( writeStream, temp );
sl@0
   891
        CleanupStack::PopAndDestroy( &temp );
sl@0
   892
        CleanupStack::PopAndDestroy( &writeStream );
sl@0
   893
        }
sl@0
   894
    else if( err == KErrNone )
sl@0
   895
	    {
sl@0
   896
	    // Close write- and open readstream
sl@0
   897
	    CleanupStack::PopAndDestroy( &writeStream );
sl@0
   898
        RFileReadStream readStream;
sl@0
   899
        User::LeaveIfError( readStream.Open( iFs, path, EFileRead ) );
sl@0
   900
        CleanupClosePushL( readStream );
sl@0
   901
sl@0
   902
        // Read entries from file to temporary array
sl@0
   903
        TUint32 count;
sl@0
   904
        TUint32 countDSRs;
sl@0
   905
        ValidateHeaderL( readStream, count, countDSRs );
sl@0
   906
        TUint32 granularity = 8;
sl@0
   907
        if (count>granularity) 
sl@0
   908
        	{
sl@0
   909
        	granularity=count;
sl@0
   910
        	}
sl@0
   911
        RFeatureServerArray temp(granularity);
sl@0
   912
        CleanupClosePushL( temp );
sl@0
   913
        for(TInt i = 0; i < count; i++)
sl@0
   914
            {
sl@0
   915
            TFeatureServerEntry entry;
sl@0
   916
            entry.InternalizeL( readStream );
sl@0
   917
            temp.AppendL( entry );
sl@0
   918
            }
sl@0
   919
        // Close read-stream and handle temp array in cleanup stack
sl@0
   920
        CleanupStack::Pop( &temp );
sl@0
   921
        CleanupStack::PopAndDestroy( &readStream );
sl@0
   922
        CleanupClosePushL( temp );
sl@0
   923
sl@0
   924
        // Set or insert a new entry in to the array
sl@0
   925
        const TUid& uid( aFeature.FeatureUid() );
sl@0
   926
        TInt index = temp.FindInOrder( uid, FindByUid );
sl@0
   927
        if( index != KErrNotFound )
sl@0
   928
            {         
sl@0
   929
            if ( aType != EFeatureFeatureDeleted )
sl@0
   930
            	{
sl@0
   931
            	temp[index].Set( aFeature);
sl@0
   932
            	}
sl@0
   933
            else
sl@0
   934
            	{
sl@0
   935
            	temp.Remove( index );
sl@0
   936
            	}
sl@0
   937
            }
sl@0
   938
        else
sl@0
   939
            {
sl@0
   940
            User::LeaveIfError( temp.InsertInOrder( aFeature, FindByUid ) );
sl@0
   941
            }
sl@0
   942
		
sl@0
   943
		//Create a Temporary File
sl@0
   944
		RFileWriteStream writeStreamTemp;
sl@0
   945
		const TPtrC KTestFile=_L("TFEATURES.DAT");
sl@0
   946
        TFileName tempPath( KCFeatMgrPrivatePath );
sl@0
   947
	    tempPath[0] = iFs.GetSystemDriveChar();
sl@0
   948
    	tempPath.Append( KTestFile );
sl@0
   949
    	User::LeaveIfError(writeStreamTemp.Replace( iFs,  tempPath, EFileWrite ));
sl@0
   950
    	CleanupClosePushL( writeStreamTemp);
sl@0
   951
    	WriteHeaderAndEntriesL( writeStreamTemp, temp );
sl@0
   952
    	writeStreamTemp.CommitL();
sl@0
   953
    	CleanupStack::PopAndDestroy(&writeStreamTemp);
sl@0
   954
        CleanupStack::PopAndDestroy( &temp );
sl@0
   955
        User::LeaveIfError(iFs.Replace(tempPath,path));		
sl@0
   956
       
sl@0
   957
	    }
sl@0
   958
    else
sl@0
   959
        {
sl@0
   960
        ERROR_LOG1( "CFeatMgrFeatureRegistry::UpdateRuntimeFeatures - err %d", err );
sl@0
   961
        User::Leave( err );
sl@0
   962
        }
sl@0
   963
    }
sl@0
   964
sl@0
   965
// -----------------------------------------------------------------------------
sl@0
   966
// CFeatMgrFeatureRegistry::WriteHeaderAndEntriesL
sl@0
   967
// -----------------------------------------------------------------------------
sl@0
   968
//  
sl@0
   969
void CFeatMgrFeatureRegistry::WriteHeaderAndEntriesL( RFileWriteStream &aStream, 
sl@0
   970
    RFeatureServerArray& aArray )
sl@0
   971
    {
sl@0
   972
    FUNC_LOG
sl@0
   973
sl@0
   974
    TInt count( aArray.Count() );
sl@0
   975
    aStream.WriteUint32L( KFileType );
sl@0
   976
    aStream.WriteUint16L( KFileVersion );
sl@0
   977
    aStream.WriteUint16L( KFileFlags );
sl@0
   978
    aStream.WriteUint32L( count );
sl@0
   979
    aStream.WriteUint32L( 0 );
sl@0
   980
    for(TInt i = 0; i < count; i++)
sl@0
   981
        {
sl@0
   982
        aArray[i].ExternalizeL( aStream );
sl@0
   983
        }
sl@0
   984
    aStream.CommitL();
sl@0
   985
    }
sl@0
   986
sl@0
   987
// -----------------------------------------------------------------------------
sl@0
   988
// CFeatMgrFeatureRegistry::MergePluginFeatures
sl@0
   989
// -----------------------------------------------------------------------------
sl@0
   990
//  
sl@0
   991
void CFeatMgrFeatureRegistry::MergePluginFeaturesL( 
sl@0
   992
        RArray<FeatureInfoCommand::TFeature>& aList )
sl@0
   993
    {
sl@0
   994
    FUNC_LOG
sl@0
   995
    
sl@0
   996
    TInt count = aList.Count();
sl@0
   997
sl@0
   998
    for ( TInt i = 0; i < count; i++ )
sl@0
   999
        {
sl@0
  1000
        const TUid uid( TUid::Uid( aList[i].iFeatureID ) );
sl@0
  1001
        TInt index = SearchFeature( uid );
sl@0
  1002
        
sl@0
  1003
        if(index != KErrNotFound)
sl@0
  1004
            {
sl@0
  1005
            if ( !IsFlagSet( index, EFeatureBlackListed ) )
sl@0
  1006
                {
sl@0
  1007
                // Update support-status bit
sl@0
  1008
                TBitFlags32 flags( iFeatureList[index].FeatureFlags() );
sl@0
  1009
                flags.Assign( EFeatureSupported, aList[i].iValue);
sl@0
  1010
                
sl@0
  1011
                // Set existing entry in array
sl@0
  1012
                TFeatureServerEntry entry( uid, flags, iFeatureList[index].FeatureData());
sl@0
  1013
                iFeatureList[index].Set(entry);
sl@0
  1014
                }
sl@0
  1015
            else
sl@0
  1016
                {
sl@0
  1017
                INFO_LOG1( "CFeatMgrFeatureRegistry::MergePluginFeatures - 0x%08x blacklisted",
sl@0
  1018
                    iFeatureList[i].FeatureUid().iUid );
sl@0
  1019
                }
sl@0
  1020
            }
sl@0
  1021
        else
sl@0
  1022
            {
sl@0
  1023
            TBitFlags32 flags;
sl@0
  1024
            flags.Assign( EFeatureSupported, aList[i].iValue);
sl@0
  1025
            // Insert new entry in array
sl@0
  1026
            TFeatureServerEntry newFeature( uid, flags, KDefaultData );
sl@0
  1027
            TInt err = iFeatureList.InsertInOrder( newFeature, FindByUid );
sl@0
  1028
            INFO_LOG2( "CFeatMgrFeatureRegistry::MergePluginFeatures - 0x%08x insert result %d",
sl@0
  1029
                newFeature.FeatureUid().iUid, err );
sl@0
  1030
            User::LeaveIfError(err);
sl@0
  1031
            }
sl@0
  1032
        }
sl@0
  1033
    }
sl@0
  1034
    
sl@0
  1035
// -----------------------------------------------------------------------------
sl@0
  1036
// CFeatMgrFeatureRegistry::MergePluginFeatures
sl@0
  1037
// -----------------------------------------------------------------------------
sl@0
  1038
//  
sl@0
  1039
void CFeatMgrFeatureRegistry::MergePluginFeaturesL( RFeatureArray& aList )
sl@0
  1040
    {
sl@0
  1041
    FUNC_LOG
sl@0
  1042
    
sl@0
  1043
    TInt count = aList.Count();
sl@0
  1044
sl@0
  1045
    for ( TInt i = 0; i < count; i++ )
sl@0
  1046
        {
sl@0
  1047
        //Check for feature flag errors
sl@0
  1048
        ValidateFeatureFlagL(aList[i].FeatureFlags()) ; 
sl@0
  1049
        const TUid uid( aList[i].FeatureUid() );
sl@0
  1050
        TInt index = SearchFeature( uid );
sl@0
  1051
        
sl@0
  1052
        if( index != KErrNotFound )
sl@0
  1053
            {
sl@0
  1054
            if ( !IsFlagSet( index, EFeatureBlackListed ) )
sl@0
  1055
                {
sl@0
  1056
                // Set existing entry in array with new info and data
sl@0
  1057
                TFeatureServerEntry entry( uid, aList[i].FeatureFlags(), aList[i].FeatureData() );
sl@0
  1058
                iFeatureList[index].Set(entry);
sl@0
  1059
                }
sl@0
  1060
            else
sl@0
  1061
                {
sl@0
  1062
                INFO_LOG1( "CFeatMgrFeatureRegistry::MergePluginFeatures - 0x%08x blacklisted",
sl@0
  1063
                    iFeatureList[i].FeatureUid().iUid );
sl@0
  1064
                }
sl@0
  1065
            }
sl@0
  1066
        else
sl@0
  1067
            {
sl@0
  1068
            // Insert new entry in array
sl@0
  1069
            TFeatureServerEntry newFeature( uid, aList[i].FeatureFlags(), aList[i].FeatureData() );
sl@0
  1070
            TInt err = iFeatureList.InsertInOrder( newFeature, FindByUid );
sl@0
  1071
            INFO_LOG2( "CFeatMgrFeatureRegistry::MergePluginFeatures - 0x%08x insert result %d",
sl@0
  1072
                newFeature.FeatureUid().iUid, err );
sl@0
  1073
            User::LeaveIfError(err);
sl@0
  1074
            }
sl@0
  1075
        }
sl@0
  1076
    }
sl@0
  1077
sl@0
  1078
// -----------------------------------------------------------------------------
sl@0
  1079
// CFeatMgrFeatureRegistry::ValidateFeatureFlag
sl@0
  1080
// Following are the rule to check err in the ROM defined feature flags
sl@0
  1081
// Rule 1) If a feature flag is blacklisted then setting any of modifiable, persisted, Un-initialised bit will be an error
sl@0
  1082
// Rule 2) If a feature flag is non blacklisted, non modifiable  setting any of Un-initialised, Persisted bit will be an error
sl@0
  1083
// -----------------------------------------------------------------------------
sl@0
  1084
//  
sl@0
  1085
   
sl@0
  1086
void CFeatMgrFeatureRegistry::ValidateFeatureFlagL(TBitFlags32 aFlags)
sl@0
  1087
	{
sl@0
  1088
	_LIT(KPanicCategory, "FEATMGR-FLAGS");
sl@0
  1089
	
sl@0
  1090
	if(!aFlags.IsSet(EFeatureRuntime)) //ROM defined feature flag error check
sl@0
  1091
		{
sl@0
  1092
		//Rule 1
sl@0
  1093
		if(aFlags.IsSet(EFeatureBlackListed) )
sl@0
  1094
	    	{
sl@0
  1095
	    	 if(aFlags.IsSet(EFeatureModifiable) || aFlags.IsSet(EFeaturePersisted) || aFlags.IsSet(EFeatureUninitialized) )
sl@0
  1096
	    	 	{
sl@0
  1097
	    	 	//error 
sl@0
  1098
	    	 	::FmgrFatalErrorL(KErrArgument, KPanicCategory, EFmpInvalidFeatureBitFlagsRule1);
sl@0
  1099
	    	  	}
sl@0
  1100
	    	}
sl@0
  1101
	    	
sl@0
  1102
	    //Rule 2    	
sl@0
  1103
	    if (!aFlags.IsSet(EFeatureModifiable))
sl@0
  1104
	       	{
sl@0
  1105
	       	if (aFlags.IsSet(EFeaturePersisted) || aFlags.IsSet(EFeatureUninitialized) )
sl@0
  1106
	       	 	{
sl@0
  1107
	        	//error 
sl@0
  1108
	       	 	::FmgrFatalErrorL(KErrArgument, KPanicCategory, EFmpInvalidFeatureBitFlagsRule2);
sl@0
  1109
	        	}	
sl@0
  1110
	        }
sl@0
  1111
		}
sl@0
  1112
	else // Runtime feature this should not be in the rom
sl@0
  1113
		{		
sl@0
  1114
		::FmgrFatalErrorL(KErrArgument, KPanicCategory, EPanicInvalidFeatureInfo);
sl@0
  1115
		}
sl@0
  1116
	}
sl@0
  1117
sl@0
  1118
/**
sl@0
  1119
 * This function is used to validate feature flags that are read from the features file on the ffs.
sl@0
  1120
 * This function is also used to validate feature flags that are modified or added with the execption of MergePluginFeatures.
sl@0
  1121
 * This validation compares the flags against a set of rules. This ffs file needs to be validate separately from the 
sl@0
  1122
 * rom file. If the rom validation method is used a panic can occur which is appropriate for checking the rom but not
sl@0
  1123
 * for the ffs.  
sl@0
  1124
 * This does not validate the dsr ranges. 
sl@0
  1125
 * The following are the rules to check for errors in the run time defined feature flags
sl@0
  1126
 * Rule 1)Blacklisting of a run-time defined feature flag is an error 
sl@0
  1127
 * Rule 2)Un-initialised feature flag should be modifiable.   
sl@0
  1128
 */
sl@0
  1129
void CFeatMgrFeatureRegistry::ValidateRuntimeFeatureFlagL(TBitFlags32 aFlags)
sl@0
  1130
	{
sl@0
  1131
	
sl@0
  1132
	//Rule 1 (Blacklisting of run-time defined feature aFlags is not allowed)
sl@0
  1133
	if(aFlags.IsSet(EFeatureBlackListed) ) 
sl@0
  1134
    	{
sl@0
  1135
		//error 
sl@0
  1136
    	User::Leave( KErrArgument );
sl@0
  1137
    	}
sl@0
  1138
    	
sl@0
  1139
    //Rule 2 (non modifiable run-time feature aFlags should initialised
sl@0
  1140
	if(!aFlags.IsSet(EFeatureModifiable) && aFlags.IsSet(EFeatureUninitialized) )
sl@0
  1141
	 	{
sl@0
  1142
	 	//error 
sl@0
  1143
	 	User::Leave( KErrArgument );
sl@0
  1144
	 	}
sl@0
  1145
	}
sl@0
  1146
sl@0
  1147
sl@0
  1148
/**
sl@0
  1149
 * After restore, some features might have changed. This function will examine
sl@0
  1150
 * the differences between the old feature set and the newly restored feature set
sl@0
  1151
 * to discover if any changes have taken place: then it will handle the required
sl@0
  1152
 * notifications for new, deleted and changed featuers.
sl@0
  1153
 */
sl@0
  1154
void CFeatMgrFeatureRegistry::HandleRestoredFeatureNotificationsL()
sl@0
  1155
	{
sl@0
  1156
	// All comparisons are between the new list iFeatureList and the old list iFeatureListBackup
sl@0
  1157
	TInt new_count = iFeatureList.Count();
sl@0
  1158
	TInt old_count = iFeatureListBackup.Count();
sl@0
  1159
	
sl@0
  1160
	// Three lists, defining the differences between the two arrays
sl@0
  1161
	RArray<TFeatureServerEntry> added;
sl@0
  1162
	RArray<TFeatureServerEntry> removed;
sl@0
  1163
	RArray<TFeatureServerEntry> changed;
sl@0
  1164
sl@0
  1165
	// Regarding the newer iFeatureList array
sl@0
  1166
	// Get the features according to the "new" iFeatureList array
sl@0
  1167
	for( TInt i=0; i < new_count; i++ )
sl@0
  1168
		{
sl@0
  1169
	    // If not set, the feature flag is a ROM or plug-in
sl@0
  1170
	    if( iFeatureList[i].FeatureFlags().IsSet(EFeatureRuntime) )
sl@0
  1171
	        {
sl@0
  1172
	        TUid uid( iFeatureList[i].FeatureUid() );
sl@0
  1173
		    TInt index = iFeatureListBackup.FindInOrder( uid, FindByUid );
sl@0
  1174
		    
sl@0
  1175
			// KErrNotFound, if no matching object can be found
sl@0
  1176
			if( KErrNotFound == index )
sl@0
  1177
				{
sl@0
  1178
				// Recently added feature
sl@0
  1179
				added.AppendL( iFeatureList[i] );
sl@0
  1180
				}
sl@0
  1181
			else
sl@0
  1182
				{
sl@0
  1183
				// Get the features in iFeatureList that have recently been altered
sl@0
  1184
		        TFeatureServerEntry old_item = iFeatureListBackup[index];
sl@0
  1185
		        TFeatureServerEntry new_item = iFeatureList[i];
sl@0
  1186
		        TUint32 old_flags = old_item.FeatureFlags().Value();
sl@0
  1187
		        TUint32 new_flags = new_item.FeatureFlags().Value();
sl@0
  1188
		        unsigned long int old_data = old_item.FeatureData();
sl@0
  1189
		        unsigned long int new_data = new_item.FeatureData();
sl@0
  1190
	        	// if any thing has changed, then add it to our list.
sl@0
  1191
		        // there is no != overload for TBitFlags32
sl@0
  1192
				if( !( old_flags == new_flags) || !( old_data == new_data) )
sl@0
  1193
					{
sl@0
  1194
					// changed in the "new" iFeatureList array
sl@0
  1195
					changed.AppendL( iFeatureList[i] );
sl@0
  1196
					}
sl@0
  1197
				}
sl@0
  1198
		
sl@0
  1199
	        } // end if ! EFeatureRuntime
sl@0
  1200
		} // end loop
sl@0
  1201
sl@0
  1202
sl@0
  1203
	// Regarding the older iFeatureListBackup array
sl@0
  1204
	// Get the features according to the "old" iFeatureListBackup array
sl@0
  1205
	for( TInt i=0; i < old_count; i++ )
sl@0
  1206
		{
sl@0
  1207
	    // If not set, the feature flag is a ROM or plug-in
sl@0
  1208
	    if( iFeatureListBackup[i].FeatureFlags().IsSet(EFeatureRuntime) )
sl@0
  1209
	        {
sl@0
  1210
	        TUid uid( iFeatureListBackup[i].FeatureUid() );
sl@0
  1211
		    TInt index = iFeatureList.FindInOrder( uid, FindByUid );
sl@0
  1212
		    
sl@0
  1213
			// KErrNotFound, if no matching object can be found
sl@0
  1214
			if( KErrNotFound == index )
sl@0
  1215
				{
sl@0
  1216
				// Recently removed feature
sl@0
  1217
				removed.AppendL( iFeatureListBackup[i] );
sl@0
  1218
				}
sl@0
  1219
			// the else has already been completed in previous loop
sl@0
  1220
		
sl@0
  1221
	        } // end if ! EFeatureRuntime
sl@0
  1222
		} // end loop
sl@0
  1223
	
sl@0
  1224
	TInt size_added 	= added.Count();
sl@0
  1225
	TInt size_changed 	= changed.Count();
sl@0
  1226
	TInt size_removed 	= removed.Count();
sl@0
  1227
sl@0
  1228
	// notify the added features
sl@0
  1229
    for( TInt i = 0; i < size_added; i++ )
sl@0
  1230
        {
sl@0
  1231
        TFeatureServerEntry entry( added[i].FeatureUid(), added[i].FeatureFlags(), added[i].FeatureData() );
sl@0
  1232
        TFeatureChangeType changeType( EFeatureFeatureCreated );
sl@0
  1233
        iObserver.HandleFeatureChange( entry, changeType );
sl@0
  1234
        }
sl@0
  1235
sl@0
  1236
	// notify the changed features
sl@0
  1237
    for( TInt i = 0; i < size_changed; i++ )
sl@0
  1238
        {
sl@0
  1239
        TFeatureServerEntry entry( changed[i].FeatureUid(), changed[i].FeatureFlags(), changed[i].FeatureData() );
sl@0
  1240
        TFeatureChangeType changeType( EFeatureStatusUpdated );
sl@0
  1241
        iObserver.HandleFeatureChange( entry, changeType );        
sl@0
  1242
        }
sl@0
  1243
sl@0
  1244
	// notify the delete features
sl@0
  1245
    for( TInt i = 0; i < size_removed; i++ )
sl@0
  1246
        {
sl@0
  1247
        TFeatureServerEntry entry( removed[i].FeatureUid(), removed[i].FeatureFlags(), removed[i].FeatureData() );
sl@0
  1248
        TFeatureChangeType changeType( EFeatureFeatureDeleted );
sl@0
  1249
        iObserver.HandleFeatureChange( entry, changeType );
sl@0
  1250
        }
sl@0
  1251
	}
sl@0
  1252
sl@0
  1253
// -----------------------------------------------------------------------------
sl@0
  1254
// CFeatMgrFeatureRegistry::SWIStart
sl@0
  1255
// -----------------------------------------------------------------------------
sl@0
  1256
//
sl@0
  1257
TInt CFeatMgrFeatureRegistry::SWIStart(TUint aSWIProcessId)
sl@0
  1258
	{
sl@0
  1259
	FUNC_LOG
sl@0
  1260
	
sl@0
  1261
	// If a previous call to SWIStart was made then return an error indicating that SWI
sl@0
  1262
	// is already running. This assures that no two exes will enable the caching
sl@0
  1263
	// mechanism at the same time.
sl@0
  1264
	if( iSWICacheFeature )
sl@0
  1265
		{
sl@0
  1266
		INFO_LOG( "CFeatMgrFeatureRegistry::SWIStart - Already in use");
sl@0
  1267
		return KErrInUse;
sl@0
  1268
		}
sl@0
  1269
	
sl@0
  1270
	RProperty propertyHndl;
sl@0
  1271
	TInt err =propertyHndl.Attach(KUidSystemCategory, KSAUidSoftwareInstallKeyValue);
sl@0
  1272
	if (KErrNone != err)
sl@0
  1273
		{
sl@0
  1274
		return err;
sl@0
  1275
		}
sl@0
  1276
	TInt val = -1;
sl@0
  1277
	err = propertyHndl.Get(val);
sl@0
  1278
	propertyHndl.Close();
sl@0
  1279
sl@0
  1280
	if( KErrNone == err )
sl@0
  1281
		{
sl@0
  1282
		// If an installation/uninstallation has started and no finishing status has been set for it
sl@0
  1283
		if( ((val&ESASwisInstall) || (val&ESASwisUninstall)) && !(val&ESASwisStatusNone) )
sl@0
  1284
			{
sl@0
  1285
			// Set a flag to tell FeatMgr that features modified from this point onwards
sl@0
  1286
		    // until a call to SWIEnd must be cached.
sl@0
  1287
		    iSWICacheFeature = ETrue;
sl@0
  1288
		    // SWI installation/uninstallation is in progress
sl@0
  1289
		    iSWIStatus = ESWIInstalling;
sl@0
  1290
		    // Set the ID of the process issuing Feature Manager commands to be cached
sl@0
  1291
			iSWIProcessId = aSWIProcessId;
sl@0
  1292
			// Start listening to P&S install property
sl@0
  1293
		    TRAP(err, iSWIListener = CSWIListener::NewL(this));
sl@0
  1294
		    if (KErrNone != err)
sl@0
  1295
		    	{
sl@0
  1296
		    	return err;
sl@0
  1297
		    	}
sl@0
  1298
		    
sl@0
  1299
		    // Start the timer to handle the case of the launched exe hanging or not calling SWIEnd
sl@0
  1300
		    // after SWIStart
sl@0
  1301
		    TRAP(err, iSWITimer = CSWITimer::NewL(TTimeIntervalMicroSeconds32(SWITIMEOUT), this));
sl@0
  1302
		    if (KErrNone != err)
sl@0
  1303
		    	{
sl@0
  1304
		    	return err;
sl@0
  1305
		    	}
sl@0
  1306
		    	
sl@0
  1307
		    INFO_LOG( "CFeatMgrFeatureRegistry::SWIStart - err KErrNone");
sl@0
  1308
		    return KErrNone;
sl@0
  1309
			}
sl@0
  1310
		}
sl@0
  1311
	INFO_LOG( "CFeatMgrFeatureRegistry::SWIStart - err KErrNotReady");
sl@0
  1312
sl@0
  1313
	return KErrNotReady;
sl@0
  1314
	}
sl@0
  1315
sl@0
  1316
// -----------------------------------------------------------------------------
sl@0
  1317
// CFeatMgrFeatureRegistry::SWIEnd
sl@0
  1318
// -----------------------------------------------------------------------------
sl@0
  1319
//
sl@0
  1320
TInt CFeatMgrFeatureRegistry::SWIEnd(TUint aSWIProcessId)
sl@0
  1321
	{
sl@0
  1322
	FUNC_LOG
sl@0
  1323
sl@0
  1324
	// reset the number of operations cached
sl@0
  1325
	iSWIOperations = 0;
sl@0
  1326
	
sl@0
  1327
	// If it is the same process that made a call to SWIStart and caching is in progress
sl@0
  1328
	if( iSWIProcessId == aSWIProcessId && iSWICacheFeature )
sl@0
  1329
		{
sl@0
  1330
		if( iSWIStatus == ESWIAborted )
sl@0
  1331
			{
sl@0
  1332
			SWIReset();
sl@0
  1333
			
sl@0
  1334
			INFO_LOG( "CFeatMgrFeatureRegistry::SWIEnd - SWIStatus = ESWIAborted");
sl@0
  1335
			return KErrNone;
sl@0
  1336
			}
sl@0
  1337
		else if( iSWIStatus == ESWIInstalling )
sl@0
  1338
			{
sl@0
  1339
			// Stop time-out
sl@0
  1340
			if( iSWITimer )
sl@0
  1341
				{
sl@0
  1342
				delete iSWITimer;
sl@0
  1343
				iSWITimer = NULL;
sl@0
  1344
				}
sl@0
  1345
			// Stop caching
sl@0
  1346
			iSWICacheFeature = EFalse;			
sl@0
  1347
			
sl@0
  1348
			TInt err = KErrGeneral;
sl@0
  1349
sl@0
  1350
			if( !iOomOccured )
sl@0
  1351
				{
sl@0
  1352
				if( iAddFeatCount>0 )
sl@0
  1353
					{
sl@0
  1354
					err = iFeatureList.Reserve(iFeatureList.Count() + iAddFeatCount);
sl@0
  1355
                    if (err == KErrNoMemory)
sl@0
  1356
                         {
sl@0
  1357
                         iSWIStatus = ESWIOutOfMemory;
sl@0
  1358
                         }
sl@0
  1359
					}
sl@0
  1360
				}
sl@0
  1361
			else
sl@0
  1362
				{
sl@0
  1363
				err = KErrNoMemory;
sl@0
  1364
				iSWIStatus = ESWIOutOfMemory;
sl@0
  1365
				}
sl@0
  1366
			
sl@0
  1367
			INFO_LOG( "CFeatMgrFeatureRegistry::SWIEnd - SWIStatus = ESWIInstalling");
sl@0
  1368
			return err;
sl@0
  1369
			}
sl@0
  1370
		}
sl@0
  1371
sl@0
  1372
	INFO_LOG( "CFeatMgrFeatureRegistry::SWIEnd - err KErrNotReady");
sl@0
  1373
	return KErrNotReady;
sl@0
  1374
	}
sl@0
  1375
sl@0
  1376
// -----------------------------------------------------------------------------
sl@0
  1377
// CFeatMgrFeatureRegistry::SWICacheCommand
sl@0
  1378
// -----------------------------------------------------------------------------
sl@0
  1379
//
sl@0
  1380
TInt CFeatMgrFeatureRegistry::SWICacheCommand(TSWIOperationCat aOptCat, TFeatureServerEntry aFeature)
sl@0
  1381
	{
sl@0
  1382
	FUNC_LOG
sl@0
  1383
	
sl@0
  1384
	TInt err;
sl@0
  1385
sl@0
  1386
	if (iSWIOperations >= MAXSWIOPS)
sl@0
  1387
		{
sl@0
  1388
		err = KErrArgument;
sl@0
  1389
		}
sl@0
  1390
	else if (iOomOccured)
sl@0
  1391
		{
sl@0
  1392
		err = KErrNoMemory;
sl@0
  1393
		}
sl@0
  1394
	else
sl@0
  1395
		{
sl@0
  1396
		if( aOptCat == ESWIAddFeat )
sl@0
  1397
			{
sl@0
  1398
			++iAddFeatCount;
sl@0
  1399
			}		
sl@0
  1400
		TSWICachedOperation operation;
sl@0
  1401
		operation.iFeatEntry = aFeature;
sl@0
  1402
		operation.iCat = aOptCat;
sl@0
  1403
    	err = iSWICachedOperations.Append(operation);
sl@0
  1404
    	if( err == KErrNoMemory)
sl@0
  1405
    		{
sl@0
  1406
    		iOomOccured = ETrue;
sl@0
  1407
    		}
sl@0
  1408
    	else if( err == KErrNone )
sl@0
  1409
    		{
sl@0
  1410
    		++iSWIOperations;
sl@0
  1411
    		}
sl@0
  1412
		}
sl@0
  1413
	return err;
sl@0
  1414
	}
sl@0
  1415
sl@0
  1416
// -----------------------------------------------------------------------------
sl@0
  1417
// CFeatMgrFeatureRegistry::CommitSWIFeatChanges
sl@0
  1418
// -----------------------------------------------------------------------------
sl@0
  1419
//
sl@0
  1420
void CFeatMgrFeatureRegistry::CommitSWIFeatChanges()
sl@0
  1421
	{
sl@0
  1422
	FUNC_LOG
sl@0
  1423
	
sl@0
  1424
	// Commit all cached features.
sl@0
  1425
	if( !iSWICacheFeature )
sl@0
  1426
		{
sl@0
  1427
		TInt count = iSWICachedOperations.Count();
sl@0
  1428
sl@0
  1429
		for( TInt i=0; i<count; ++i )
sl@0
  1430
			{
sl@0
  1431
			TSWIOperationCat optCat = iSWICachedOperations[i].iCat;
sl@0
  1432
	
sl@0
  1433
			switch(optCat)
sl@0
  1434
				{
sl@0
  1435
			case ESWIAddFeat:
sl@0
  1436
				{
sl@0
  1437
				AddFeature( iSWICachedOperations[i].iFeatEntry, 0 );
sl@0
  1438
				}
sl@0
  1439
				break;
sl@0
  1440
			case ESWIDeleteFeat:
sl@0
  1441
				{
sl@0
  1442
				DeleteFeature( iSWICachedOperations[i].iFeatEntry.FeatureUid() );
sl@0
  1443
				}
sl@0
  1444
				break;
sl@0
  1445
			case ESWISetFeatAndData:
sl@0
  1446
				{
sl@0
  1447
				TUint32 data = iSWICachedOperations[i].iFeatEntry.FeatureData();				
sl@0
  1448
				SetFeature( iSWICachedOperations[i].iFeatEntry.FeatureUid(),
sl@0
  1449
							iSWICachedOperations[i].iFeatEntry.FeatureFlags().Value(),
sl@0
  1450
							&data);
sl@0
  1451
		    	}
sl@0
  1452
				break;
sl@0
  1453
			case ESWISetFeatData:
sl@0
  1454
				{
sl@0
  1455
				TUint32 data = iSWICachedOperations[i].iFeatEntry.FeatureData();
sl@0
  1456
				SetFeature(iSWICachedOperations[i].iFeatEntry.FeatureUid(),
sl@0
  1457
							EFeatureSupportUntouch,&data);
sl@0
  1458
				}
sl@0
  1459
				break;
sl@0
  1460
            case ESWISetFeat:
sl@0
  1461
                {
sl@0
  1462
                SetFeature( iSWICachedOperations[i].iFeatEntry.FeatureUid(),
sl@0
  1463
                            iSWICachedOperations[i].iFeatEntry.FeatureFlags().Value(),
sl@0
  1464
                            NULL);
sl@0
  1465
                }
sl@0
  1466
                break;
sl@0
  1467
			default:
sl@0
  1468
				break;
sl@0
  1469
				};
sl@0
  1470
			}
sl@0
  1471
		
sl@0
  1472
		INFO_LOG( "CFeatMgrFeatureRegistry::CommitSWIFeatChanges - Committing completed");
sl@0
  1473
		}
sl@0
  1474
	SWIReset();
sl@0
  1475
	}
sl@0
  1476
sl@0
  1477
// -----------------------------------------------------------------------------
sl@0
  1478
// CFeatMgrFeatureRegistry::SWIReset
sl@0
  1479
// -----------------------------------------------------------------------------
sl@0
  1480
//
sl@0
  1481
void CFeatMgrFeatureRegistry::SWIReset()
sl@0
  1482
	{
sl@0
  1483
	// Clear cached-features array
sl@0
  1484
	iSWICachedOperations.Close();
sl@0
  1485
sl@0
  1486
	// Reset caching flag
sl@0
  1487
	iSWICacheFeature = EFalse;
sl@0
  1488
	
sl@0
  1489
	// Reset SWI process Id
sl@0
  1490
	iSWIProcessId = 1;
sl@0
  1491
	
sl@0
  1492
	// Reset SWI completion status
sl@0
  1493
	iSWIStatus = ESWIComplete;
sl@0
  1494
sl@0
  1495
	// Reset the check for "out of memory" condition
sl@0
  1496
	iOomOccured = EFalse;
sl@0
  1497
	
sl@0
  1498
	// Reset the operations counter
sl@0
  1499
	iSWIOperations = 0;
sl@0
  1500
	
sl@0
  1501
	// No need to listen to P&S insall property any more
sl@0
  1502
	delete iSWIListener;
sl@0
  1503
	iSWIListener = NULL;
sl@0
  1504
	
sl@0
  1505
	// Stop time-out
sl@0
  1506
	if( iSWITimer )
sl@0
  1507
		{
sl@0
  1508
		delete iSWITimer;
sl@0
  1509
		iSWITimer = NULL;
sl@0
  1510
		}
sl@0
  1511
	}
sl@0
  1512
sl@0
  1513
// -----------------------------------------------------------------------------
sl@0
  1514
// CFeatMgrFeatureRegistry::SWIAborted
sl@0
  1515
// -----------------------------------------------------------------------------
sl@0
  1516
//
sl@0
  1517
void CFeatMgrFeatureRegistry::SWIAborted()
sl@0
  1518
	{
sl@0
  1519
	FUNC_LOG
sl@0
  1520
	
sl@0
  1521
	// If abort occured before SWIEnd is called
sl@0
  1522
	if( iSWICacheFeature )
sl@0
  1523
		{
sl@0
  1524
		INFO_LOG( "CFeatMgrFeatureRegistry::SWIAborted - Abort occured before SWIEnd was called");
sl@0
  1525
		iSWIStatus = ESWIAborted;
sl@0
  1526
		}
sl@0
  1527
	else
sl@0
  1528
		{
sl@0
  1529
		INFO_LOG( "CFeatMgrFeatureRegistry::SWIAborted - Abort occured after SWIEnd was called");
sl@0
  1530
		SWIReset();
sl@0
  1531
		}
sl@0
  1532
	}
sl@0
  1533
sl@0
  1534
// -----------------------------------------------------------------------------
sl@0
  1535
// CFeatMgrFeatureRegistry::SWITimedOut
sl@0
  1536
// -----------------------------------------------------------------------------
sl@0
  1537
//
sl@0
  1538
void CFeatMgrFeatureRegistry::SWITimedOut()
sl@0
  1539
	{
sl@0
  1540
	FUNC_LOG
sl@0
  1541
	
sl@0
  1542
	if( iSWICacheFeature )
sl@0
  1543
		{
sl@0
  1544
		INFO_LOG( "CFeatMgrFeatureRegistry::SWITimedOut - Timeout expired");
sl@0
  1545
		SWIReset();
sl@0
  1546
		}
sl@0
  1547
	}
sl@0
  1548
sl@0
  1549
// -----------------------------------------------------------------------------
sl@0
  1550
// CFeatMgrFeatureRegistry::SWICacheStarted
sl@0
  1551
// -----------------------------------------------------------------------------
sl@0
  1552
//
sl@0
  1553
TBool CFeatMgrFeatureRegistry::SWICacheStarted()
sl@0
  1554
	{
sl@0
  1555
	return iSWICacheFeature;
sl@0
  1556
	}
sl@0
  1557
sl@0
  1558
// -----------------------------------------------------------------------------
sl@0
  1559
// CFeatMgrFeatureRegistry::SWICacheStatus
sl@0
  1560
// -----------------------------------------------------------------------------
sl@0
  1561
//
sl@0
  1562
TBool CFeatMgrFeatureRegistry::SWICacheStatusOOM()
sl@0
  1563
	{
sl@0
  1564
	if( iSWIStatus == ESWIOutOfMemory )
sl@0
  1565
		{
sl@0
  1566
		return ETrue;
sl@0
  1567
		}
sl@0
  1568
	return EFalse;
sl@0
  1569
	}
sl@0
  1570
	
sl@0
  1571
//  End of File