os/persistentdata/featuremgmt/featuremgr/src/serverexe/featmgrfeatureregistry.cpp
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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
20 #include <ecom/ecom.h>
23 #include <e32property.h>
25 #include "featmgrconfiguration.h"
26 #include "featmgrfeatureregistry.h"
27 #include "featmgrserver.h"
28 #include "featmgrdebug.h"
31 #define SWITIMEOUT 15000000
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\\" );
40 _LIT( KZFeaturesDir, "Z:\\Private\\10205054\\" );
41 _LIT( KCFeatMgrPrivatePath, "?:\\Private\\10205054\\" );
42 #endif // EXTENDED_FEATURE_MANAGER_TEST
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 );
52 // ============================= LOCAL FUNCTIONS ===============================
54 // ============================ MEMBER FUNCTIONS ===============================
56 // -----------------------------------------------------------------------------
57 // CFeatMgrFeatureRegistry::CFeatMgrFeatureRegistry
58 // -----------------------------------------------------------------------------
60 CFeatMgrFeatureRegistry::CFeatMgrFeatureRegistry( RFs& aFs,
61 MFeatMgrRegistryObserver& aObserver )
63 iObserver( aObserver ),
65 iSWICacheFeature( EFalse ),
66 iSWIStatus( ESWIComplete ),
72 // -----------------------------------------------------------------------------
73 // CFeatMgrFeatureRegistry::ConstructL
74 // Symbian 2nd phase constructor can leave.
75 // -----------------------------------------------------------------------------
77 void CFeatMgrFeatureRegistry::ConstructL()
81 // -----------------------------------------------------------------------------
82 // CFeatMgrFeatureRegistry::NewL
83 // Two-phased constructor.
84 // -----------------------------------------------------------------------------
86 CFeatMgrFeatureRegistry* CFeatMgrFeatureRegistry::NewL( RFs& aFs,
87 MFeatMgrRegistryObserver& aObserver )
89 CFeatMgrFeatureRegistry* self =
90 new( ELeave ) CFeatMgrFeatureRegistry( aFs, aObserver );
92 CleanupStack::PushL( self );
94 CleanupStack::Pop( self );
99 // -----------------------------------------------------------------------------
101 // -----------------------------------------------------------------------------
103 CFeatMgrFeatureRegistry::~CFeatMgrFeatureRegistry()
107 iFeatureList.Close();
109 iFeatureListBackup.Close();
110 iSWICachedOperations.Close();
122 // -----------------------------------------------------------------------------
123 // CFeatMgrFeatureRegistry::IsFeatureSupported
124 // -----------------------------------------------------------------------------
126 TInt CFeatMgrFeatureRegistry::IsFeatureSupported( TFeatureServerEntry& aFeature )
128 TInt err( KErrNotFound );
129 const TInt index = SearchFeature( aFeature.FeatureUid() );
131 if ( index == KErrNotFound )
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++ )
138 if( (uid.iUid >= iRangeList[i].iLowUid.iUid) &&
139 (uid.iUid <= iRangeList[i].iHighUid.iUid) )
141 TBitFlags32 flags( 0 );
142 flags.Assign( EFeatureSupported, KFeatureSupported );
143 TFeatureServerEntry entry( aFeature.FeatureUid(), flags, KDefaultData );
145 err = KFeatureSupported;
150 else if( IsFlagSet( index, EFeatureUninitialized ) )
152 // Supported status bit is not taken into account if feature not yet initialized
155 else if ( (index < iFeatureList.Count()) && IsFlagSet( index, EFeatureSupported ) )
157 TBitFlags32 flags = iFeatureList[index].FeatureFlags();
158 flags.Assign( EFeatureSupported, KFeatureSupported );
159 TUint32 data = iFeatureList[index].FeatureData();
160 TFeatureServerEntry entry( aFeature.FeatureUid(), flags, data );
162 err = KFeatureSupported;
166 TBitFlags32 flags = iFeatureList[index].FeatureFlags();
167 flags.Assign( EFeatureSupported, KFeatureUnsupported );
168 TUint32 data = iFeatureList[index].FeatureData();
169 TFeatureServerEntry entry( aFeature.FeatureUid(), flags, data );
171 err = KFeatureUnsupported;
177 // -----------------------------------------------------------------------------
178 // CFeatMgrFeatureRegistry::AddFeature
179 // -----------------------------------------------------------------------------
181 TInt CFeatMgrFeatureRegistry::AddFeature( TFeatureServerEntry& aFeature, TUint aPrcId )
183 TInt err( KErrAccessDenied );
185 if( iSWIProcessId == aPrcId && iSWICacheFeature )
187 err = SWICacheCommand(ESWIAddFeat, aFeature);
191 const TInt index = SearchFeature( aFeature.FeatureUid() );
193 if ( index == KErrNotFound )
195 TBitFlags32 flags( aFeature.FeatureFlags() );
196 flags.Set( EFeatureRuntime );
198 //Check the feature falg is valid
199 TRAP(err,ValidateRuntimeFeatureFlagL(flags));
203 TFeatureServerEntry entry( aFeature.FeatureUid(), flags, aFeature.FeatureData() );
204 err = iFeatureList.InsertInOrder( entry, FindByUid );
205 if ( err == KErrNone )
207 TFeatureChangeType changeType( EFeatureFeatureCreated );
208 err = HandleChange( entry, changeType );
213 err = KErrAlreadyExists;
216 INFO_LOG("CFeatMgrFeatureRegistry::AddFeature - Features directly stored in registry");
219 LOG_IF_ERROR1( err, "CFeatMgrFeatureRegistry::AddFeature - result %d", err );
224 // -----------------------------------------------------------------------------
225 // CFeatMgrFeatureRegistry::DeleteFeature
226 // -----------------------------------------------------------------------------
228 TInt CFeatMgrFeatureRegistry::DeleteFeature( TUid aFeature, TUint aPrcId )
230 TInt err( KErrAccessDenied );
232 if( iSWIProcessId == aPrcId && iSWICacheFeature )
234 err = SWICacheCommand(ESWIDeleteFeat, aFeature);
238 // Check if the feature is runtime
239 TInt index = SearchFeature( aFeature );
240 if ( index == KErrNotFound )
244 if ( !iFeatureList[index].FeatureFlags().IsSet(EFeatureRuntime) )
246 return KErrAccessDenied;
249 TFeatureServerEntry entry = iFeatureList[index];
251 iFeatureList.Remove( index );
252 TFeatureChangeType changeType( EFeatureFeatureDeleted );
253 err = HandleChange( entry, changeType );
255 INFO_LOG("CFeatMgrFeatureRegistry::DeleteFeature - Features deleted directly from registry");
258 LOG_IF_ERROR1( err, "CFeatMgrFeatureRegistry::DeleteFeature - result %d", err );
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 // -----------------------------------------------------------------------------
269 TInt CFeatMgrFeatureRegistry::SetFeature( TUid aFeature, TInt aEnable, const TUint32 *aData, TUint aPrcId )
273 TInt err( KErrNone );
275 if( iSWIProcessId == aPrcId && iSWICacheFeature )
277 TBitFlags32 flags(0);
278 flags.Assign( EFeatureSupported, aEnable );
285 TFeatureServerEntry entry( aFeature, flags, data );
287 // If aData is not null, we want to change the user data too
288 // otherwise only change the feature status.
291 err = SWICacheCommand(ESWISetFeatAndData, entry);
295 err = SWICacheCommand(ESWISetFeat, entry);
302 // Validate feature exists and is modifiable
303 err = ValidateFeature( aFeature, index );
305 if ( err != KErrNone )
310 if ( (index >= 0 && index < iFeatureList.Count()) )
312 TBitFlags32 flags = iFeatureList[index].FeatureFlags();
313 TUint32 data = iFeatureList[index].FeatureData();
314 TFeatureChangeType changeType( EFeatureStatusUpdated );
316 // Update "supported" info according to request
317 if( (aEnable == EFeatureSupportEnable) || (aEnable == EFeatureSupportDisable) )
319 INFO_LOG1( "CFeatMgrFeatureRegistry::SetFeature() - aEnable %d", aEnable );
320 flags.Assign( EFeatureSupported, aEnable );
322 // When setting feature, always unset "uninitialized" bit
323 flags.Assign( EFeatureUninitialized, 0 );
325 // Update data whenever applied
328 INFO_LOG1( "CFeatMgrFeatureRegistry::SetFeature() - aData %d", aData );
331 if( aEnable == EFeatureSupportUntouch )
333 changeType = EFeatureDataUpdated;
337 changeType = EFeatureStatusDataUpdated;
341 TFeatureServerEntry entry( aFeature, flags, data );
342 //Check if the feature will actually change
343 if(iFeatureList[index].FeatureFlags() == flags && iFeatureList[index].FeatureData() == data )
345 //No change were made, set change type to EFeatureNoChange
346 changeType = EFeatureNoChange;
350 // Set the feature entry in list with updated information
351 iFeatureList[index].Set(entry);
354 err = HandleChange( entry, changeType );
358 LOG_IF_ERROR1( err, "CFeatMgrFeatureRegistry::SetFeature - result %d", err );
363 // -----------------------------------------------------------------------------
364 // CFeatMgrFeatureRegistry::HandleChange
365 // -----------------------------------------------------------------------------
367 TInt CFeatMgrFeatureRegistry::HandleChange( TFeatureServerEntry& aFeature,
368 TFeatureChangeType aType )
372 TInt err( KErrNone );
374 // Update feature file, when feature is specified as persisted.
375 if ( aFeature.FeatureFlags().IsSet( EFeaturePersisted ) )
377 TRAP( err, UpdateRuntimeFeaturesFileL( aFeature, aType ) );
378 LOG_IF_ERROR1( err, "CFeatMgrFeatureRegistry::HandleChange - update error %d", err );
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 )
385 // It is also questionnable whether we should suppress notification in case file
387 // if ( err == KErrNone )
388 iObserver.HandleFeatureChange( aFeature, aType );
393 // -----------------------------------------------------------------------------
394 // CFeatMgrFeatureRegistry::ValidateFeature
395 // -----------------------------------------------------------------------------
397 TInt CFeatMgrFeatureRegistry::ValidateFeature( TUid aFeature, TInt &aIndex )
399 TInt err( KErrNone );
401 aIndex = SearchFeature( aFeature );
403 if ( aIndex == KErrNotFound )
407 else if ( !IsFlagSet( aIndex, EFeatureModifiable ) )
409 err = KErrAccessDenied;
415 // -----------------------------------------------------------------------------
416 // CFeatMgrFeatureRegistry::FindByUid
417 // Returns Zero if UIDs do match.
418 // -----------------------------------------------------------------------------
420 TInt CFeatMgrFeatureRegistry::FindByUid( const TUid *aFeature,
421 const TFeatureServerEntry& aItem )
423 if ( aFeature->iUid < aItem.FeatureUid().iUid )
427 else if ( aFeature->iUid > aItem.FeatureUid().iUid )
435 // -----------------------------------------------------------------------------
436 // CFeatMgrFeatureRegistry::FindByUid
437 // Returns Zero if UIDs do match.
438 // -----------------------------------------------------------------------------
440 TInt CFeatMgrFeatureRegistry::FindByUid( const TFeatureServerEntry& aFeature,
441 const TFeatureServerEntry& aItem )
443 if ( aFeature.FeatureUid().iUid < aItem.FeatureUid().iUid )
447 else if ( aFeature.FeatureUid().iUid > aItem.FeatureUid().iUid )
455 // -----------------------------------------------------------------------------
456 // CFeatMgrFeatureRegistry::SearchFeature
457 // -----------------------------------------------------------------------------
459 TInt CFeatMgrFeatureRegistry::SearchFeature( TUid aFeature )
461 const TUid& uid( aFeature );
462 return iFeatureList.FindInOrder( uid, FindByUid );
465 // -----------------------------------------------------------------------------
466 // CFeatMgrFeatureRegistry::IsFlagSet
467 // -----------------------------------------------------------------------------
469 TBool CFeatMgrFeatureRegistry::IsFlagSet( TInt aIndex, TFeatureFlags aFlag )
471 TBool isSet( EFalse );
472 if( aIndex < iFeatureList.Count() )
474 isSet = iFeatureList[aIndex].FeatureFlags().IsSet(aFlag);
480 // -----------------------------------------------------------------------------
481 // CFeatMgrFeatureRegistry::SupportedFeatures
482 // -----------------------------------------------------------------------------
484 void CFeatMgrFeatureRegistry::SupportedFeaturesL( RFeatureUidArray& aSupportedFeatures )
488 TInt count = iFeatureList.Count();
490 for ( TInt i = 0; i < count; i++ )
492 if( IsFlagSet( i, EFeatureSupported) )
494 aSupportedFeatures.AppendL( iFeatureList[i].FeatureUid() );
499 // -----------------------------------------------------------------------------
500 // CFeatMgrFeatureRegistry::NumberOfSupportedFeatures
501 // -----------------------------------------------------------------------------
503 TInt CFeatMgrFeatureRegistry::NumberOfSupportedFeatures()
507 TInt count = iFeatureList.Count();
508 TInt countSupported(0);
510 for ( TInt i = 0; i < count; i++ )
512 if( IsFlagSet( i, EFeatureSupported) )
518 return countSupported;
521 // -----------------------------------------------------------------------------
523 void CFeatMgrFeatureRegistry::ResetFeaturesL()
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++)
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]);
538 // destroy the feature list
539 iFeatureList.Reset();
540 iFeatureList.Close();
547 * Get the fully qualified path and filename to the features.dat
550 TFileName CFeatMgrFeatureRegistry::GetFeaturesFilePathAndName( void )
552 TFileName path( KCFeatMgrPrivatePath );
554 path[0] = iFs.GetSystemDriveChar();
555 path.Append( KCRuntimeFeaturesFileName );
560 // CFeatMgrFeatureRegistry::ReadFeatureFilesL
561 // Reads platform and product feature files.
562 // -----------------------------------------------------------------------------
564 void CFeatMgrFeatureRegistry::ReadFeatureFilesL()
568 // Read feature files from Z
569 ReadFilesFromDirL( KZFeaturesDir );
571 //check that there is at least one DSR
572 if (!iRangeList.Count())
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);
580 // -----------------------------------------------------------------------------
581 // CFeatMgrFeatureRegistry::ReadFilesFromDirL
582 // -----------------------------------------------------------------------------
584 void CFeatMgrFeatureRegistry::ReadFilesFromDirL( const TDesC& aDirName )
586 _LIT(KPanicCategory, "FEATMGR-READFILE");
589 TInt err( KErrNone );
591 err = iFs.GetDir( aDirName, KEntryAttNormal, ESortByName, dir );
592 CleanupStack::PushL( dir );
594 if( err == KErrNone )
596 err = ReadFiles( aDirName, dir );
597 if ( err != KErrNone )
599 ERROR_LOG1( "CFeatMgrFeatureRegistry::ReadFilesFromDirL() - err %d ", err );
603 else if( err == KErrPathNotFound )
605 ::FmgrFatalErrorL(err, KPanicCategory, EPanicNoFeatureFiles);
609 ERROR_LOG1( "CFeatMgrFeatureRegistry::ReadFilesFromDirL() - err %d ", err );
613 CleanupStack::PopAndDestroy( dir );
616 // -----------------------------------------------------------------------------
617 // CFeatMgrFeatureRegistry::ReadRuntimeFeaturesL
618 // -----------------------------------------------------------------------------
620 void CFeatMgrFeatureRegistry::ReadRuntimeFeaturesL( TBool &aFeaturesReady )
622 TFileName path( KCFeatMgrPrivatePath );
623 path[0] = iFs.GetSystemDriveChar();
624 path.Append( KCRuntimeFeaturesFileName );
626 TInt err( KErrNone );
627 TRAP( err, ReadFileL( path ) );
629 if ((err == KErrCorrupt) || (err == KErrArgument))
631 User::LeaveIfError(iFs.Delete(path));
632 aFeaturesReady = ETrue;
634 else if ( err != KErrNone && err != KErrNotFound && err != KErrPathNotFound )
636 ERROR_LOG1( "CFeatMgrFeatureRegistry::ReadRuntimeFeatures - ReadFileL returned err %d", err );
641 aFeaturesReady = ETrue;
645 // -----------------------------------------------------------------------------
646 // CFeatMgrFeatureRegistry::ReadFiles
647 // -----------------------------------------------------------------------------
649 TInt CFeatMgrFeatureRegistry::ReadFiles( const TDesC& aPath, CDir* aDir )
651 TInt fileCount = aDir->Count();
653 TInt err( KErrNotFound );
655 for ( TInt file = 0; file < fileCount; file++ )
657 TInt match = (*aDir)[file].iName.MatchC( KZFeaturesFileNameMatch );
658 if( match != KErrNotFound )
660 fileName.Copy(aPath);
661 fileName.Append((*aDir)[file].iName);
663 INFO_LOG1( "CFeatMgrFeatureRegistry::ReadFiles - file: %S", &fileName );
664 TRAP( err, ReadFileL( fileName ) );
665 LOG_IF_ERROR2( err, "CFeatMgrFeatureRegistry::ReadFiles - file: %S, err %d",
668 // Return error if reading of any feature file fails.
669 if( err != KErrNone )
679 // -----------------------------------------------------------------------------
680 // CFeatMgrFeatureRegistry::ReadFileL
681 // -----------------------------------------------------------------------------
683 void CFeatMgrFeatureRegistry::ReadFileL( const TDesC& aFullPath )
688 RFileReadStream readStream;
690 // Open the file and attach to stream
691 User::LeaveIfError( readStream.Open( iFs, aFullPath, EFileRead ) );
692 CleanupClosePushL( readStream );
695 //Validate the header
696 ValidateHeaderL( readStream, count, countDSRs );
698 RArray<TFeatureServerEntry> tempFeatureArray;
699 CleanupClosePushL( tempFeatureArray );
701 //Find the directory that the feature file is contained in.
702 TFileName dirName(aFullPath);
704 dirName.Delete((dirName.LocateReverse(toFind)+1), dirName.Length() );
705 TBool runtimeFile = EFalse;
706 if (dirName.Compare(KZFeaturesDir) != 0) //Location of the feature file.
711 tempFeatureArray.ReserveL(count);
713 for(TInt i = 0; i < count; i++)
715 TFeatureServerEntry entry;
716 entry.InternalizeL( readStream );
718 //Check for feature flag errors
719 TBitFlags32 flags = entry.FeatureFlags();
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.
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.
729 if (SearchFeature( entry.FeatureUid() ) == KErrNotFound )// Check to see if the feature has been read in previously from the rom.
731 User::Leave(KErrCorrupt);
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
735 ValidateRuntimeFeatureFlagL(flags);
738 else //Flag set the feature is runtime this is then validated as normal
740 ValidateRuntimeFeatureFlagL(flags);
744 else //File is not as runtime file.
746 ValidateFeatureFlagL(flags);
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) )
755 tempFeatureArray.InsertL( entry, i);
758 // Reserve memory if list still empty
759 if( !iFeatureList.Count() )
761 iFeatureList.ReserveL( tempFeatureArray.Count() );
764 // Read features from temp array
765 for(TInt i = 0; i < tempFeatureArray.Count(); i++)
767 TFeatureServerEntry entry = tempFeatureArray[i];
769 TInt index = SearchFeature( entry.FeatureUid() );
771 if( index == KErrNotFound)
773 iFeatureList.InsertInOrderL( entry, FindByUid );
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 ) )
782 iFeatureList[index].Set(entry);
787 // Reserve memory if DSR list still empty
788 if( !iRangeList.Count() )
790 iRangeList.ReserveL( countDSRs );
793 // Read default supported ranges from file
794 for(TInt i = 0; i < countDSRs; i++)
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 )
802 ERROR_LOG( "CFeatMgrFeatureRegistry::ReadFileL - invalid supported range" );
803 iRangeList.Remove( i );
804 User::Leave( KErrCorrupt );
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++)
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() );
818 count = iRangeList.Count();
819 INFO_LOG1( "CFeatMgrFeatureRegistry::ReadFileL - supported ranges: %d", count );
820 for(TInt i = 0; i < count; i++)
822 INFO_LOG2( "CFeatMgrFeatureRegistry::ReadFileL - low 0x%08x, high 0x%08x",
823 iRangeList[i].iLowUid, iRangeList[i].iHighUid );
827 CleanupStack::PopAndDestroy( &tempFeatureArray);
828 CleanupStack::PopAndDestroy( &readStream );
832 // -----------------------------------------------------------------------------
833 // CFeatMgrFeatureRegistry::ValidateHeaderL
834 // -----------------------------------------------------------------------------
836 void CFeatMgrFeatureRegistry::ValidateHeaderL( RFileReadStream &aStream,
837 TUint32& aCount, TUint32& aCountDSRs )
841 TUint32 identifier = aStream.ReadUint32L();
842 TUint16 fileVersion = aStream.ReadUint16L();
843 TUint16 fileFlags = aStream.ReadUint16L();
844 aCount = aStream.ReadUint32L();
845 aCountDSRs = aStream.ReadUint32L();
847 // Carry out simple verification of file content
848 if((identifier != KFileType) || fileVersion != KFileVersion ||
849 fileFlags != KFileFlags )
851 User::Leave( KErrCorrupt );
855 // -----------------------------------------------------------------------------
856 // CFeatMgrFeatureRegistry::UpdateRuntimeFeaturesFileL
857 // -----------------------------------------------------------------------------
859 void CFeatMgrFeatureRegistry::UpdateRuntimeFeaturesFileL( TFeatureServerEntry& aFeature,
860 TFeatureChangeType aType )
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 );
871 err = writeStream.Open( iFs, path, EFileWrite );
872 CleanupClosePushL( writeStream );
874 if( err == KErrPathNotFound || err == KErrNotFound )
876 // Create folder and file.
877 if ( err == KErrPathNotFound )
879 path = KCFeatMgrPrivatePath;
880 path[0] = iFs.GetSystemDriveChar();
881 User::LeaveIfError( iFs.MkDirAll( path ) );
882 path.Append( KCRuntimeFeaturesFileName );
884 User::LeaveIfError( writeStream.Create( iFs, path, EFileWrite ) );
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 );
894 else if( err == KErrNone )
896 // Close write- and open readstream
897 CleanupStack::PopAndDestroy( &writeStream );
898 RFileReadStream readStream;
899 User::LeaveIfError( readStream.Open( iFs, path, EFileRead ) );
900 CleanupClosePushL( readStream );
902 // Read entries from file to temporary array
905 ValidateHeaderL( readStream, count, countDSRs );
906 TUint32 granularity = 8;
907 if (count>granularity)
911 RFeatureServerArray temp(granularity);
912 CleanupClosePushL( temp );
913 for(TInt i = 0; i < count; i++)
915 TFeatureServerEntry entry;
916 entry.InternalizeL( readStream );
917 temp.AppendL( entry );
919 // Close read-stream and handle temp array in cleanup stack
920 CleanupStack::Pop( &temp );
921 CleanupStack::PopAndDestroy( &readStream );
922 CleanupClosePushL( temp );
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 )
929 if ( aType != EFeatureFeatureDeleted )
931 temp[index].Set( aFeature);
935 temp.Remove( index );
940 User::LeaveIfError( temp.InsertInOrder( aFeature, FindByUid ) );
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));
960 ERROR_LOG1( "CFeatMgrFeatureRegistry::UpdateRuntimeFeatures - err %d", err );
965 // -----------------------------------------------------------------------------
966 // CFeatMgrFeatureRegistry::WriteHeaderAndEntriesL
967 // -----------------------------------------------------------------------------
969 void CFeatMgrFeatureRegistry::WriteHeaderAndEntriesL( RFileWriteStream &aStream,
970 RFeatureServerArray& aArray )
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++)
982 aArray[i].ExternalizeL( aStream );
987 // -----------------------------------------------------------------------------
988 // CFeatMgrFeatureRegistry::MergePluginFeatures
989 // -----------------------------------------------------------------------------
991 void CFeatMgrFeatureRegistry::MergePluginFeaturesL(
992 RArray<FeatureInfoCommand::TFeature>& aList )
996 TInt count = aList.Count();
998 for ( TInt i = 0; i < count; i++ )
1000 const TUid uid( TUid::Uid( aList[i].iFeatureID ) );
1001 TInt index = SearchFeature( uid );
1003 if(index != KErrNotFound)
1005 if ( !IsFlagSet( index, EFeatureBlackListed ) )
1007 // Update support-status bit
1008 TBitFlags32 flags( iFeatureList[index].FeatureFlags() );
1009 flags.Assign( EFeatureSupported, aList[i].iValue);
1011 // Set existing entry in array
1012 TFeatureServerEntry entry( uid, flags, iFeatureList[index].FeatureData());
1013 iFeatureList[index].Set(entry);
1017 INFO_LOG1( "CFeatMgrFeatureRegistry::MergePluginFeatures - 0x%08x blacklisted",
1018 iFeatureList[i].FeatureUid().iUid );
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);
1035 // -----------------------------------------------------------------------------
1036 // CFeatMgrFeatureRegistry::MergePluginFeatures
1037 // -----------------------------------------------------------------------------
1039 void CFeatMgrFeatureRegistry::MergePluginFeaturesL( RFeatureArray& aList )
1043 TInt count = aList.Count();
1045 for ( TInt i = 0; i < count; i++ )
1047 //Check for feature flag errors
1048 ValidateFeatureFlagL(aList[i].FeatureFlags()) ;
1049 const TUid uid( aList[i].FeatureUid() );
1050 TInt index = SearchFeature( uid );
1052 if( index != KErrNotFound )
1054 if ( !IsFlagSet( index, EFeatureBlackListed ) )
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);
1062 INFO_LOG1( "CFeatMgrFeatureRegistry::MergePluginFeatures - 0x%08x blacklisted",
1063 iFeatureList[i].FeatureUid().iUid );
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);
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 // -----------------------------------------------------------------------------
1086 void CFeatMgrFeatureRegistry::ValidateFeatureFlagL(TBitFlags32 aFlags)
1088 _LIT(KPanicCategory, "FEATMGR-FLAGS");
1090 if(!aFlags.IsSet(EFeatureRuntime)) //ROM defined feature flag error check
1093 if(aFlags.IsSet(EFeatureBlackListed) )
1095 if(aFlags.IsSet(EFeatureModifiable) || aFlags.IsSet(EFeaturePersisted) || aFlags.IsSet(EFeatureUninitialized) )
1098 ::FmgrFatalErrorL(KErrArgument, KPanicCategory, EFmpInvalidFeatureBitFlagsRule1);
1103 if (!aFlags.IsSet(EFeatureModifiable))
1105 if (aFlags.IsSet(EFeaturePersisted) || aFlags.IsSet(EFeatureUninitialized) )
1108 ::FmgrFatalErrorL(KErrArgument, KPanicCategory, EFmpInvalidFeatureBitFlagsRule2);
1112 else // Runtime feature this should not be in the rom
1114 ::FmgrFatalErrorL(KErrArgument, KPanicCategory, EPanicInvalidFeatureInfo);
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
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.
1129 void CFeatMgrFeatureRegistry::ValidateRuntimeFeatureFlagL(TBitFlags32 aFlags)
1132 //Rule 1 (Blacklisting of run-time defined feature aFlags is not allowed)
1133 if(aFlags.IsSet(EFeatureBlackListed) )
1136 User::Leave( KErrArgument );
1139 //Rule 2 (non modifiable run-time feature aFlags should initialised
1140 if(!aFlags.IsSet(EFeatureModifiable) && aFlags.IsSet(EFeatureUninitialized) )
1143 User::Leave( KErrArgument );
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.
1154 void CFeatMgrFeatureRegistry::HandleRestoredFeatureNotificationsL()
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();
1160 // Three lists, defining the differences between the two arrays
1161 RArray<TFeatureServerEntry> added;
1162 RArray<TFeatureServerEntry> removed;
1163 RArray<TFeatureServerEntry> changed;
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++ )
1169 // If not set, the feature flag is a ROM or plug-in
1170 if( iFeatureList[i].FeatureFlags().IsSet(EFeatureRuntime) )
1172 TUid uid( iFeatureList[i].FeatureUid() );
1173 TInt index = iFeatureListBackup.FindInOrder( uid, FindByUid );
1175 // KErrNotFound, if no matching object can be found
1176 if( KErrNotFound == index )
1178 // Recently added feature
1179 added.AppendL( iFeatureList[i] );
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) )
1194 // changed in the "new" iFeatureList array
1195 changed.AppendL( iFeatureList[i] );
1199 } // end if ! EFeatureRuntime
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++ )
1207 // If not set, the feature flag is a ROM or plug-in
1208 if( iFeatureListBackup[i].FeatureFlags().IsSet(EFeatureRuntime) )
1210 TUid uid( iFeatureListBackup[i].FeatureUid() );
1211 TInt index = iFeatureList.FindInOrder( uid, FindByUid );
1213 // KErrNotFound, if no matching object can be found
1214 if( KErrNotFound == index )
1216 // Recently removed feature
1217 removed.AppendL( iFeatureListBackup[i] );
1219 // the else has already been completed in previous loop
1221 } // end if ! EFeatureRuntime
1224 TInt size_added = added.Count();
1225 TInt size_changed = changed.Count();
1226 TInt size_removed = removed.Count();
1228 // notify the added features
1229 for( TInt i = 0; i < size_added; i++ )
1231 TFeatureServerEntry entry( added[i].FeatureUid(), added[i].FeatureFlags(), added[i].FeatureData() );
1232 TFeatureChangeType changeType( EFeatureFeatureCreated );
1233 iObserver.HandleFeatureChange( entry, changeType );
1236 // notify the changed features
1237 for( TInt i = 0; i < size_changed; i++ )
1239 TFeatureServerEntry entry( changed[i].FeatureUid(), changed[i].FeatureFlags(), changed[i].FeatureData() );
1240 TFeatureChangeType changeType( EFeatureStatusUpdated );
1241 iObserver.HandleFeatureChange( entry, changeType );
1244 // notify the delete features
1245 for( TInt i = 0; i < size_removed; i++ )
1247 TFeatureServerEntry entry( removed[i].FeatureUid(), removed[i].FeatureFlags(), removed[i].FeatureData() );
1248 TFeatureChangeType changeType( EFeatureFeatureDeleted );
1249 iObserver.HandleFeatureChange( entry, changeType );
1253 // -----------------------------------------------------------------------------
1254 // CFeatMgrFeatureRegistry::SWIStart
1255 // -----------------------------------------------------------------------------
1257 TInt CFeatMgrFeatureRegistry::SWIStart(TUint aSWIProcessId)
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 )
1266 INFO_LOG( "CFeatMgrFeatureRegistry::SWIStart - Already in use");
1270 RProperty propertyHndl;
1271 TInt err =propertyHndl.Attach(KUidSystemCategory, KSAUidSoftwareInstallKeyValue);
1272 if (KErrNone != err)
1277 err = propertyHndl.Get(val);
1278 propertyHndl.Close();
1280 if( KErrNone == err )
1282 // If an installation/uninstallation has started and no finishing status has been set for it
1283 if( ((val&ESASwisInstall) || (val&ESASwisUninstall)) && !(val&ESASwisStatusNone) )
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)
1299 // Start the timer to handle the case of the launched exe hanging or not calling SWIEnd
1301 TRAP(err, iSWITimer = CSWITimer::NewL(TTimeIntervalMicroSeconds32(SWITIMEOUT), this));
1302 if (KErrNone != err)
1307 INFO_LOG( "CFeatMgrFeatureRegistry::SWIStart - err KErrNone");
1311 INFO_LOG( "CFeatMgrFeatureRegistry::SWIStart - err KErrNotReady");
1313 return KErrNotReady;
1316 // -----------------------------------------------------------------------------
1317 // CFeatMgrFeatureRegistry::SWIEnd
1318 // -----------------------------------------------------------------------------
1320 TInt CFeatMgrFeatureRegistry::SWIEnd(TUint aSWIProcessId)
1324 // reset the number of operations cached
1327 // If it is the same process that made a call to SWIStart and caching is in progress
1328 if( iSWIProcessId == aSWIProcessId && iSWICacheFeature )
1330 if( iSWIStatus == ESWIAborted )
1334 INFO_LOG( "CFeatMgrFeatureRegistry::SWIEnd - SWIStatus = ESWIAborted");
1337 else if( iSWIStatus == ESWIInstalling )
1346 iSWICacheFeature = EFalse;
1348 TInt err = KErrGeneral;
1352 if( iAddFeatCount>0 )
1354 err = iFeatureList.Reserve(iFeatureList.Count() + iAddFeatCount);
1355 if (err == KErrNoMemory)
1357 iSWIStatus = ESWIOutOfMemory;
1364 iSWIStatus = ESWIOutOfMemory;
1367 INFO_LOG( "CFeatMgrFeatureRegistry::SWIEnd - SWIStatus = ESWIInstalling");
1372 INFO_LOG( "CFeatMgrFeatureRegistry::SWIEnd - err KErrNotReady");
1373 return KErrNotReady;
1376 // -----------------------------------------------------------------------------
1377 // CFeatMgrFeatureRegistry::SWICacheCommand
1378 // -----------------------------------------------------------------------------
1380 TInt CFeatMgrFeatureRegistry::SWICacheCommand(TSWIOperationCat aOptCat, TFeatureServerEntry aFeature)
1386 if (iSWIOperations >= MAXSWIOPS)
1390 else if (iOomOccured)
1396 if( aOptCat == ESWIAddFeat )
1400 TSWICachedOperation operation;
1401 operation.iFeatEntry = aFeature;
1402 operation.iCat = aOptCat;
1403 err = iSWICachedOperations.Append(operation);
1404 if( err == KErrNoMemory)
1406 iOomOccured = ETrue;
1408 else if( err == KErrNone )
1416 // -----------------------------------------------------------------------------
1417 // CFeatMgrFeatureRegistry::CommitSWIFeatChanges
1418 // -----------------------------------------------------------------------------
1420 void CFeatMgrFeatureRegistry::CommitSWIFeatChanges()
1424 // Commit all cached features.
1425 if( !iSWICacheFeature )
1427 TInt count = iSWICachedOperations.Count();
1429 for( TInt i=0; i<count; ++i )
1431 TSWIOperationCat optCat = iSWICachedOperations[i].iCat;
1437 AddFeature( iSWICachedOperations[i].iFeatEntry, 0 );
1440 case ESWIDeleteFeat:
1442 DeleteFeature( iSWICachedOperations[i].iFeatEntry.FeatureUid() );
1445 case ESWISetFeatAndData:
1447 TUint32 data = iSWICachedOperations[i].iFeatEntry.FeatureData();
1448 SetFeature( iSWICachedOperations[i].iFeatEntry.FeatureUid(),
1449 iSWICachedOperations[i].iFeatEntry.FeatureFlags().Value(),
1453 case ESWISetFeatData:
1455 TUint32 data = iSWICachedOperations[i].iFeatEntry.FeatureData();
1456 SetFeature(iSWICachedOperations[i].iFeatEntry.FeatureUid(),
1457 EFeatureSupportUntouch,&data);
1462 SetFeature( iSWICachedOperations[i].iFeatEntry.FeatureUid(),
1463 iSWICachedOperations[i].iFeatEntry.FeatureFlags().Value(),
1472 INFO_LOG( "CFeatMgrFeatureRegistry::CommitSWIFeatChanges - Committing completed");
1477 // -----------------------------------------------------------------------------
1478 // CFeatMgrFeatureRegistry::SWIReset
1479 // -----------------------------------------------------------------------------
1481 void CFeatMgrFeatureRegistry::SWIReset()
1483 // Clear cached-features array
1484 iSWICachedOperations.Close();
1486 // Reset caching flag
1487 iSWICacheFeature = EFalse;
1489 // Reset SWI process Id
1492 // Reset SWI completion status
1493 iSWIStatus = ESWIComplete;
1495 // Reset the check for "out of memory" condition
1496 iOomOccured = EFalse;
1498 // Reset the operations counter
1501 // No need to listen to P&S insall property any more
1502 delete iSWIListener;
1503 iSWIListener = NULL;
1513 // -----------------------------------------------------------------------------
1514 // CFeatMgrFeatureRegistry::SWIAborted
1515 // -----------------------------------------------------------------------------
1517 void CFeatMgrFeatureRegistry::SWIAborted()
1521 // If abort occured before SWIEnd is called
1522 if( iSWICacheFeature )
1524 INFO_LOG( "CFeatMgrFeatureRegistry::SWIAborted - Abort occured before SWIEnd was called");
1525 iSWIStatus = ESWIAborted;
1529 INFO_LOG( "CFeatMgrFeatureRegistry::SWIAborted - Abort occured after SWIEnd was called");
1534 // -----------------------------------------------------------------------------
1535 // CFeatMgrFeatureRegistry::SWITimedOut
1536 // -----------------------------------------------------------------------------
1538 void CFeatMgrFeatureRegistry::SWITimedOut()
1542 if( iSWICacheFeature )
1544 INFO_LOG( "CFeatMgrFeatureRegistry::SWITimedOut - Timeout expired");
1549 // -----------------------------------------------------------------------------
1550 // CFeatMgrFeatureRegistry::SWICacheStarted
1551 // -----------------------------------------------------------------------------
1553 TBool CFeatMgrFeatureRegistry::SWICacheStarted()
1555 return iSWICacheFeature;
1558 // -----------------------------------------------------------------------------
1559 // CFeatMgrFeatureRegistry::SWICacheStatus
1560 // -----------------------------------------------------------------------------
1562 TBool CFeatMgrFeatureRegistry::SWICacheStatusOOM()
1564 if( iSWIStatus == ESWIOutOfMemory )