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