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>
22 #include "featmgrconfiguration.h"
23 #include "featmgrserver.h"
24 #include "featmgrsession.h"
25 #include "featmgrpluginhandler.h"
26 #include "featmgrdebug.h"
27 #include "featmgrsecuritypolicy.h"
31 _LIT(KPanicCategory, "FeatMgrServer");
33 // ============================ MEMBER FUNCTIONS ===============================
35 // -----------------------------------------------------------------------------
36 // CFeatMgrServer::CFeatMgrServer
37 // C++ default constructor can NOT contain any code, that
39 // -----------------------------------------------------------------------------
41 CFeatMgrServer::CFeatMgrServer( const TInt aPriority, const TServerType aType )
43 CPolicyServer( aPriority, KFeatMgrPlatSecPolicy, aType ), iBurState(this),
44 iBURInProgress(EFalse),
45 iPluginsReady(EFalse),
46 iPluginsDeleted( EFalse ),
47 iPendingRequests( ETrue ),
48 iFeaturesReady(EFalse)
54 // -----------------------------------------------------------------------------
55 // CFeatMgrServer::ConstructL
56 // Symbian 2nd phase constructor can leave.
57 // -----------------------------------------------------------------------------
59 void CFeatMgrServer::ConstructL()
63 #ifndef EXTENDED_FEATURE_MANAGER_TEST
64 // Set server process system critical
65 User::SetProcessCritical(User::ESystemCritical);
67 iShutdown.ConstructL();
68 // ensure the server still exits even if the 1st client fails to connect
69 if( !iShutdown.IsActive() )
75 // Add server to active scheduler
76 StartL( KServerProcessName );
78 // Connect to file server.
79 User::LeaveIfError( iFs.Connect() );
82 iBackupNotification = CBaBackupSessionWrapper::NewL();
85 TFileName temp( iRegistry->GetFeaturesFilePathAndName() );
86 iBackupNotification->RegisterFileL( temp, *this );
89 iRegistry = CFeatMgrFeatureRegistry::NewL( iFs, *this );
91 TRAPD(err,iRegistry->ReadFeatureFilesL());
95 ERROR_LOG( "CFeatMgrServer::ConstructL() - no feature files found in ROM - going to panic");
96 ::FmgrFatalErrorL(err, KPanicCategory, EPanicNoFeatureFiles);
99 ERROR_LOG( "CFeatMgrServer::ConstructL() - feature information in ROM is invalid - going to panic");
100 ::FmgrFatalErrorL(err, KPanicCategory, EPanicInvalidFeatureInfo);
103 User::LeaveIfError(err);
107 // List and load plugins and feature info.
108 TBool found = LoadPluginsL();
112 // Timer for deleting plugins and for comparing feature lists.
113 iTimer = CFeatMgrTimer::NewL( *this );
117 iRegistry->ReadRuntimeFeaturesL( iFeaturesReady );
121 // -----------------------------------------------------------------------------
122 // CFeatMgrServer::NewLC
123 // Two-phased constructor.
124 // -----------------------------------------------------------------------------
126 CFeatMgrServer* CFeatMgrServer::NewLC(const TInt aPriority)
130 CFeatMgrServer* self = new( ELeave ) CFeatMgrServer(aPriority, EUnsharableSessions);
132 CleanupStack::PushL( self );
138 #ifdef EXTENDED_FEATURE_MANAGER_TEST
139 // -----------------------------------------------------------------------------
140 // A new session is being created
141 // Cancel the shutdown timer if it was running
142 // -----------------------------------------------------------------------------
144 void CFeatMgrServer::AddSession()
152 // -----------------------------------------------------------------------------
153 // A session is being destroyed
154 // Start the shutdown timer if it is the last session.
155 // -----------------------------------------------------------------------------
157 void CFeatMgrServer::DropSession()
161 if (--iSessionCount==0)
163 if( !iShutdown.IsActive() )
170 // ---------------------------------------------------------------------------
172 // ---------------------------------------------------------------------------
173 inline CShutdown::CShutdown()
176 CActiveScheduler::Add(this);
179 inline void CShutdown::ConstructL()
181 CTimer::ConstructL();
184 inline void CShutdown::Start()
187 After(KShutdownTimeout);
190 void CShutdown::RunL()
193 CActiveScheduler::Stop();
196 #endif // EXTENDED_FEATURE_MANAGER_TEST
198 // -----------------------------------------------------------------------------
200 // -----------------------------------------------------------------------------
202 CFeatMgrServer::~CFeatMgrServer()
206 // Close all open sessions
207 CFeatMgrSession* session=NULL;
208 iSessionIter.SetToFirst();
210 while( (session = static_cast<CFeatMgrSession*>(iSessionIter++)) != NULL )
215 // Delete plugin handlers
216 if ( !iPluginsDeleted )
218 iPendingRequests = EFalse; // No need to serve pending requests
219 iFeaturesReady = ETrue; // No need to read files anymore
226 // De register Backup and Restore and cleanup memory
227 if( iBackupNotification )
229 TFileName temp( iRegistry->GetFeaturesFilePathAndName() );
230 iBackupNotification->DeregisterFile( temp );
231 delete iBackupNotification;
232 iBackupNotification = NULL;
240 // -----------------------------------------------------------------------------
241 // CFeatMgrServer::NewSessionL
242 // Creates a new CSession2
243 // -----------------------------------------------------------------------------
245 CSession2* CFeatMgrServer::NewSessionL( const TVersion& aVersion,
246 const RMessage2& /*aMessage*/ ) const
250 if ( !User::QueryVersionSupported( TVersion( KServerVersionMajor,
252 KServerVersionBuild ),
255 User::Leave( KErrNotSupported );
258 CSession2* session = CFeatMgrSession::NewL(*const_cast<CFeatMgrServer*>(this), *iRegistry );
263 // -----------------------------------------------------------------------------
264 // CFeatMgrServer::TimerFired
265 // Handles timer firing.
266 // -----------------------------------------------------------------------------
268 void CFeatMgrServer::TimerFired()
272 // Delete plugin handlers. Plugin handler deletes the plugin.
273 if ( !iPluginsDeleted )
279 // -----------------------------------------------------------------------------
280 // CFeatMgrServer::HandleFeatureChange()
281 // -----------------------------------------------------------------------------
283 void CFeatMgrServer::HandleFeatureChange( TFeatureServerEntry& aFeature, TFeatureChangeType aType )
287 CFeatMgrSession* session=NULL;
288 TDblQueIter<CSession2> local_iSessionIter = iSessionIter;
289 local_iSessionIter.SetToFirst();
291 while( (session = static_cast<CFeatMgrSession*>(local_iSessionIter++)) != NULL )
293 session->ServiceNotifications( aFeature, aType );
297 // ---------------------------------------------------------------------------
298 // CFeatMgrServer::ResetAndDestroyArray
299 // ---------------------------------------------------------------------------
301 static void ResetAndDestroyArray( TAny* aPtr )
303 RImplInfoPtrArray* array = static_cast< RImplInfoPtrArray* >( aPtr );
304 array->ResetAndDestroy();
308 // -----------------------------------------------------------------------------
309 // CFeatMgrServer::LoadPluginsL()
310 // -----------------------------------------------------------------------------
312 TBool CFeatMgrServer::LoadPluginsL()
316 RImplInfoPtrArray implInfoArray;
319 // Use ECom to read information about existing interface implementations
320 #ifndef EXTENDED_FEATURE_MANAGER_TEST
321 _LIT8( KResolverString, "" );
323 _LIT8( KResolverString, "efmtestplugin" ); //In test server we only want test plugins.
325 TEComResolverParams resolverParams;
326 resolverParams.SetDataType (KResolverString);
327 resolverParams.SetWildcardMatch (ETrue);
329 TCleanupItem cleanupItem( ResetAndDestroyArray, &implInfoArray );
330 CleanupStack::PushL( cleanupItem );
332 TIMESTAMP( "CFeatMgrServer::LoadPluginsL - ListImplementationsL start: " )
333 REComSession::ListImplementationsL( KFeatureInfoPluginInterfaceUid,
335 #ifndef EXTENDED_FEATURE_MANAGER_TEST
339 TIMESTAMP( "CFeatMgrServer::LoadPluginsL - ListImplementationsL end: " )
341 // Check if any plugin was found.
342 TInt count = implInfoArray.Count();
345 iPluginsReady = ETrue; // Plugins not found.
346 INFO_LOG1("CFeatMgrServer::LoadPluginsL - interfaceUid.iUid == 0x%x, return plugins not found", KFeatureInfoPluginInterfaceUid);
351 for(TInt i=0;i<count;++i)
353 CFeatMgrPluginHandler* pluginHandler = NULL;
354 TRAPD(err, pluginHandler = CFeatMgrPluginHandler::NewL(implInfoArray[i]->ImplementationUid(), *this));
357 CleanupStack::PushL(pluginHandler);
358 TRAP(err, pluginHandler->SendCommandL(FeatureInfoCommand::ELoadFeatureInfoCmdId));
359 if(err == KErrNotSupported)
361 TRAP(err, pluginHandler->SendCommandL(FeatureInfoCommand::ELoadEnhancedFeatureInfoCmdId));
364 if(err == KErrNoMemory)
368 else if(err != KErrNone)
370 ERROR_LOG2("CFeatMgrServer::LoadPluginsL() - implementationUid: 0x%x, error: %d - going to panic", implInfoArray[i]->ImplementationUid(), err);
371 ::FmgrFatalErrorL(err, KPanicCategory, EPanicLoadPluginError);
373 // Add information of the plugin to the plugin list. Set all plugins as not ready initially.
374 SFeatMgrPluginInfo plugin;
375 plugin.iPluginHandler = pluginHandler;
376 plugin.iPluginReady = EFalse;
377 User::LeaveIfError(iPluginList.Append(plugin));
378 CleanupStack::Pop(pluginHandler);
380 INFO_LOG1("CFeatMgrServer::LoadPluginsL - interfaceUid.iUid == 0x%x, return plugins found", KFeatureInfoPluginInterfaceUid);
384 CleanupStack::PopAndDestroy(&implInfoArray);
390 // -----------------------------------------------------------------------------
391 // CFeatMgrServer::PluginsReady()
392 // -----------------------------------------------------------------------------
394 TBool CFeatMgrServer::PluginsReady() const
397 return(iPluginsReady);
400 // -----------------------------------------------------------------------------
401 // CFeatMgrServer::BackupIsInProgress()
402 // -----------------------------------------------------------------------------
403 TBool CFeatMgrServer::BURIsInProgress() const
406 return(iBURInProgress);
409 // -----------------------------------------------------------------------------
410 // CFeatMgrServer::ServicePendingRequests()
411 // -----------------------------------------------------------------------------
413 void CFeatMgrServer::ServicePendingRequests()
417 CFeatMgrSession* session=NULL;
418 iSessionIter.SetToFirst();
420 while( (session = static_cast<CFeatMgrSession*>(iSessionIter++)) != NULL )
422 TRAPD( err, session->ServicePendingRequestsL() );
423 LOG_IF_ERROR1( err, "CFeatMgrServer::ServicePendingRequests - err %d", err );
426 iPendingRequests = EFalse;
429 // -----------------------------------------------------------------------------
430 // CFeatMgrServer::DeletePlugins()
431 // -----------------------------------------------------------------------------
433 void CFeatMgrServer::DeletePlugins()
437 // Set iPluginsReady to ETrue because plugins will be deleted.
438 iPluginsReady = ETrue;
439 TInt err( KErrNone );
441 // Load runtimefeatures.txt if not loaded yet.
442 if ( !iFeaturesReady )
444 TRAP( err, iRegistry->ReadRuntimeFeaturesL( iFeaturesReady ) );
446 LOG_IF_ERROR1( err, "CFeatMgrServer::DeletePlugins() - ReadRuntimeFeaturesL err %d", err );
449 // Service pending requests before deleting plugins if not serviced yet.
450 if ( iPendingRequests )
452 ServicePendingRequests();
455 // Delete plugin handlers
456 TInt count = iPluginList.Count();
458 for ( TInt i = 0 ; i < count; i++ )
460 delete iPluginList[i].iPluginHandler;
461 iPluginList[i].iPluginHandler = NULL;
464 // Reset list of plugins
467 // Signal final closure of ecom session
468 REComSession::FinalClose();
470 // All plugin handlers deleted
471 iPluginsDeleted = ETrue;
474 // -----------------------------------------------------------------------------
475 // CFeatMgrServer::FeatureInfoL()
476 // -----------------------------------------------------------------------------
478 void CFeatMgrServer::FeatureInfoL( RArray<FeatureInfoCommand::TFeature>& aFeatureList,
479 CFeatMgrPluginHandler* aPluginHandler )
483 // Save feature info received from the plugin
484 TInt pluginCount = iPluginList.Count();
486 for ( TInt i = 0; i < pluginCount; i ++ )
488 if ( iPluginList[i].iPluginHandler == aPluginHandler )
490 iRegistry->MergePluginFeaturesL( aFeatureList );
491 // Send command to load enhanced feature info
492 TRAPD( err, iPluginList[i].iPluginHandler->SendCommandL(
493 FeatureInfoCommand::ELoadEnhancedFeatureInfoCmdId ) );
494 // Panic if error sth else than not supported
495 if ( err != KErrNone && err != KErrNotSupported )
497 ERROR_LOG1( "CFeatMgrServer::FeatureInfoL() - panicing due error %d", err );
498 ::FmgrFatalErrorL(err, KPanicCategory, EPanicLoadPluginError);
500 // At this point we have simple feature supported by the plugin.
501 // If no enhanced feature is supported, but a simple one is, then
502 // the plugin is ready to be used. Otherwise continue to add the
503 // enhanced features.
504 else if( err == KErrNotSupported )
506 iPluginList[i].iPluginReady = ETrue;
511 INFO_LOG( "CFeatMgrServer::FeatureInfo - unknown plugin handler") ;
515 CheckPluginsReadyL();
518 // -----------------------------------------------------------------------------
519 // CFeatMgrServer::FeatureInfoL()
520 // -----------------------------------------------------------------------------
522 void CFeatMgrServer::FeatureInfoL( RFeatureArray& aFeatureList,
523 CFeatMgrPluginHandler* aPluginHandler )
527 // Save feature info received from the plugin
528 TInt pluginCount = iPluginList.Count();
530 for ( TInt i = 0; i < pluginCount; i ++ )
532 if ( iPluginList[i].iPluginHandler == aPluginHandler )
534 iRegistry->MergePluginFeaturesL( aFeatureList );
535 // Send another command if something left to process
536 iPluginList[i].iPluginReady = ETrue;
540 INFO_LOG( "CFeatMgrServer::FeatureInfo - unknown plugin handler") ;
544 CheckPluginsReadyL();
547 // -----------------------------------------------------------------------------
548 // CFeatMgrServer::CheckPluginsReadyL()
549 // -----------------------------------------------------------------------------
551 void CFeatMgrServer::CheckPluginsReadyL()
555 // If all plugins has responsed iPluginsReady is ETrue
556 // else iPluginsReady is EFalse;
557 TInt pluginCount = iPluginList.Count();
559 for ( TInt i = 0; i < pluginCount; i ++ )
561 if ( iPluginList[i].iPluginReady )
563 iPluginsReady = ETrue;
567 iPluginsReady = EFalse;
572 // If plugins ready, read runtime features and call ServicePendingRequests of sessions.
575 // Load runtimefeatures.txt
576 if ( !iFeaturesReady )
578 iRegistry->ReadRuntimeFeaturesL( iFeaturesReady );
581 if ( iPendingRequests )
583 ServicePendingRequests();
589 * Implementation of the pure virtual function.
591 void CFeatMgrServer::HandleBackupOperationEventL( const TBackupOperationAttributes& /* aBackupOperationAttributes */ )
597 * There is a flag state variable being passed into Feature Manager as type MBackupObserver::TFileLockFlags.
598 * Only certain combinations of these flags are valid for Feature Manager and if the combination is
599 * not valid, then because there is no way of returning an error it will just set the internal state
600 * machine into an "error" state.
601 * Given our current state and our goto state (i.e. where we are at the moment and where we want to goto)
602 * the state machine checks that this is a valid path in our state machine and then perform the correct
603 * operations to get us to the next valid state. It also calls the correct Feature manager functions to
604 * accomplish this operation.
606 void CFeatMgrServer::ChangeFileLockL( const TDesC& /* aFileName */, TFileLockFlags aFlags )
608 // There is only one file we are concerned with, so
609 // don't check the filename matches
610 // Default TOperationType type to ENone (we are ignoring this value)
611 TBackupOperationAttributes attribs( aFlags, ENone );
612 iBurState.BUROperationL( attribs );
619 * Generic return function, to return the state of the machine back to normal.
620 * This can only occur from certain states.
622 BURStatus CFeatMgrServer::Goto_NormalState( BURStatus /* aCurrent */ )
624 return EFeatMgrBURState_None; // all roads lead to normal state
628 * This function handles error recovery from an invalid state. It will return the feature
629 * maneger back to normal, in both it's internal state machine state, and it's internal
632 BURStatus CFeatMgrServer::Goto_ErrorState( BURStatus aCurrent )
634 BURStatus aNewState = EFeatMgrBURState_None; // fail safe: all roads lead to normal state
638 case( EFeatMgrBURState_BackupStarted ) :
639 iBURInProgress = EFalse;
640 ServicePendingRequests();
642 case( EFeatMgrBURState_RestoreStarted ) :
643 iPluginsReady = EFalse;
644 iPluginsDeleted = EFalse;
645 iPendingRequests = ETrue;
646 iFeaturesReady = EFalse;
647 // re-read the new features;
648 TRAP_IGNORE(ClearFeaturesL() );
649 TRAP_IGNORE(LoadFeaturesL() );
651 iPluginsReady = ETrue;
652 // commit the pending change requests
653 ServicePendingRequests();
655 case( EFeatMgrBURState_BackupEnded ) :
656 case( EFeatMgrBURState_RestoreEnded ) :
662 aNewState = EFeatMgrBURState_Error; // state++
666 BURStatus CFeatMgrServer::Goto_StartBackupState( BURStatus /* aCurrent */ )
668 BURStatus aNewState = EFeatMgrBURState_BackupStarted; // state++
669 iBURInProgress = ETrue;
673 BURStatus CFeatMgrServer::Goto_EndBackupState( BURStatus /* aCurrent */ )
675 BURStatus aNewState = EFeatMgrBURState_BackupEnded; // state++
677 iBURInProgress = EFalse;
678 ServicePendingRequests();
679 // no error from ServicePendingRequests() is possible
684 BURStatus CFeatMgrServer::Goto_StartRestoreState( BURStatus /* aCurrent */ )
686 // remarkably like Goto_StartBackupState
687 BURStatus aNewState = EFeatMgrBURState_RestoreStarted; // state++
688 iBURInProgress = ETrue;
692 BURStatus CFeatMgrServer::Goto_EndRestoreState( BURStatus /* aCurrent */ )
694 BURStatus aNewState = EFeatMgrBURState_Error; // fail safe
695 TInt err( KErrNone );
696 iBURInProgress = EFalse;
697 iPluginsReady = EFalse;
698 iPluginsDeleted = EFalse;
699 iPendingRequests = ETrue;
700 iFeaturesReady = EFalse;
702 // re-read the new features
703 // Only call the next function if the previous one was
704 // successfully completed.
705 TRAP(err, ClearFeaturesL() );
708 TRAP(err, LoadFeaturesL() );
713 TRAP(err, HandleRestoredNotificationsL() );
719 iPluginsReady = ETrue;
721 // commit the pending change requests
722 ServicePendingRequests();
724 // Change state machine
725 aNewState = EFeatMgrBURState_RestoreEnded;
730 aNewState = EFeatMgrBURState_Error;
737 * This function will load features from files and plugins
739 void CFeatMgrServer::LoadFeaturesL( void )
742 TRAPD( err, iRegistry->ReadFeatureFilesL() );
743 if( KErrNotFound == err)
745 ERROR_LOG( "CFeatMgrServer::ConstructL() & CallReadFeatureFilesL() - no feature files found in ROM - going to panic");
746 ::FmgrFatalErrorL(err, KPanicCategory, EPanicNoFeatureFiles);
750 User::LeaveIfError(err);
754 // List and load plugins and feature info.
755 TBool found = LoadPluginsL();
759 // Timer for deleting plugins and for comparing feature lists.
760 iTimer = CFeatMgrTimer::NewL( *this );
764 TRAPD( err, iRegistry->ReadRuntimeFeaturesL( iFeaturesReady ) );
765 if( KErrNotFound == err)
767 ::FmgrFatalErrorL(err, KPanicCategory, EPanicNoFeatureFiles);
771 User::LeaveIfError(err);
780 * Discover which features have changes wrt the restore operation
782 void CFeatMgrServer::HandleRestoredNotificationsL( void )
784 iRegistry->HandleRestoredFeatureNotificationsL();
788 * This function will clear features from RAM
790 void CFeatMgrServer::ClearFeaturesL( void )
792 // Delete plugin handlers
793 if ( !iPluginsDeleted )
795 // Delete plugin handlers
796 TInt count = iPluginList.Count();
797 for ( TInt i = 0 ; i < count; i++ )
799 delete iPluginList[i].iPluginHandler;
800 iPluginList[i].iPluginHandler = NULL;
803 // Reset list of plugins
807 // All plugin handlers deleted
808 iPluginsDeleted = EFalse;
811 if( NULL != iRegistry )
813 iRegistry->ResetFeaturesL();