os/persistentdata/persistentstorage/centralrepository/cenrepsrv/install.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 1997-2009 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 // Implements installed file notification
    15 // 
    16 //
    17 
    18 /**
    19  @internalComponent
    20 */
    21 
    22 #include "install.h"
    23 #include "srvrepos_noc.h"
    24 #include "srvres.h"
    25 #include "srvparams.h"
    26 #include "cachemgr.h"
    27 
    28 CCentRepSWIWatcher* CCentRepSWIWatcher::NewL(RFs& aFs)
    29 	{
    30 	CCentRepSWIWatcher* self = new(ELeave) CCentRepSWIWatcher(aFs);
    31 	CleanupStack::PushL(self);
    32 	self->ConstructL();
    33 	CleanupStack::Pop(self);
    34 	return self;
    35 	}
    36 	
    37 void CCentRepSWIWatcher::ConstructL()
    38 	{
    39 	// Attach to SWI property
    40 	User::LeaveIfError(iSWIKey.Attach( KUidSystemCategory, KSAUidSoftwareInstallKeyValue));
    41 	
    42 	// Initialise SWI operation and status
    43 	TInt swiProperty;
    44 	const TInt error = iSWIKey.Get(KUidSystemCategory, KSAUidSoftwareInstallKeyValue, swiProperty);
    45 	
    46 	if (error == KErrNone)
    47 		{
    48 		iSWIOperation=swiProperty & KSASwisOperationMask;
    49 		iSWIStatus= swiProperty & KSASwisOperationStatusMask;
    50 		}
    51 	else if (error != KErrNotFound)
    52 		{
    53 		User::LeaveIfError(error);
    54 		}
    55 	
    56 	// Get last saved contents of install directory
    57 	GetInstallDirL();
    58 
    59 	// Do any actions required by pre server start-up SWI activity
    60 	TRAP_IGNORE(FindChangedEntriesL(ETrue));
    61 	}
    62 	
    63 CCentRepSWIWatcher::CCentRepSWIWatcher(RFs& aFs) :
    64  	CActive(EPriorityStandard), 
    65  	iInstallDir(*TServerResources::iInstallDirectory),
    66  	iFs(aFs),
    67  	iSWIOperation( ESASwisNone),
    68 	iSWIStatus( ESASwisStatusSuccess)
    69 	{
    70  	CActiveScheduler::Add(this);
    71 	}
    72 	
    73 CCentRepSWIWatcher::~CCentRepSWIWatcher()
    74 	{
    75 	Cancel();
    76 	if(iSWIKey.Handle() != KNullHandle)
    77 		{
    78 		iSWIKey.Cancel();
    79 		iSWIKey.Close();
    80 		}
    81 	iInstallEntryArray.ResetAndDestroy();
    82 	iCurrentInstallDirEntries.ResetAndDestroy();
    83 	}
    84 
    85 void CCentRepSWIWatcher::Start()
    86 	{
    87 	if(IsActive())
    88 		return;
    89 	
    90 	NotifyChange();
    91 	}
    92 	
    93  void CCentRepSWIWatcher::NotifyChange()
    94  	{
    95 	
    96 	// Register for P&S of SWI flag
    97 	iSWIKey.Subscribe(iStatus);
    98 	
    99 	SetActive();
   100 	}
   101 
   102 void CCentRepSWIWatcher::RunL()
   103 	{	
   104 	NotifyChange();
   105 		
   106 	// Get SWI Key 
   107 	TInt swiProperty;
   108 	User::LeaveIfError(iSWIKey.Get(KUidSystemCategory, KSAUidSoftwareInstallKeyValue, swiProperty));
   109 	
   110 	HandleSWIEventL(swiProperty);
   111 	}
   112 	
   113 void CCentRepSWIWatcher::HandleSWIEventL(TInt aSWIProperty)
   114 	{	
   115 	iSWIOperation=aSWIProperty & KSASwisOperationMask;
   116 	iSWIStatus=   aSWIProperty & KSASwisOperationStatusMask;
   117 
   118 	// Need to handle successful and aborted install/uninstall and successful restore
   119 	// Can't handle aborted restore
   120 	switch(iSWIOperation)
   121 		{
   122 		case ESASwisNone:
   123 			break;
   124 		case ESASwisInstall:
   125 		case ESASwisUninstall:
   126 			if(iSWIStatus==ESASwisStatusSuccess)
   127 				{
   128 				// Handle SWI events
   129 				FindChangedEntriesL();
   130 				}
   131 			else if(iSWIStatus==ESASwisStatusAborted)
   132 				{
   133 				// Update directory to reset timestamps
   134 				ReadInstallDirL(iInstallEntryArray);
   135 				SaveInstallDirL();
   136 				}
   137 			break;
   138 		case ESASwisRestore:
   139 			break;
   140 		default:
   141 			break;
   142 		}
   143 	}
   144 	
   145 // Catch leaves so they don't stop the server
   146 TInt CCentRepSWIWatcher::RunError( TInt aError)
   147 	{
   148 	static_cast<void>(aError);
   149 	
   150 	RDebug::Print(_L("Run error %d"), aError);
   151 	
   152 	// Reinitialise directory list
   153 	iInstallEntryArray.ResetAndDestroy();
   154 	
   155 	if(!IsActive())
   156 		{
   157 		NotifyChange();
   158 		}
   159 		
   160 	// Renable cache activity in case of errors during SWI events 
   161 	TServerResources::iCacheManager->EnableCache();
   162 		
   163 	return KErrNone;
   164 	}
   165 	
   166 void CCentRepSWIWatcher::DoCancel()
   167 	{
   168 	// Cancel subscription to SW P&S flag
   169 	iSWIKey.Cancel();
   170 	iSWIKey.Close();
   171 	}
   172 
   173 void CCentRepSWIWatcher::ReadInstallDirL(RPointerArray<CInstallEntry> &aEntryArray)
   174 	{
   175 	RDir dir;
   176     CleanupClosePushL(dir);
   177     
   178 	//Empty contents of directory
   179 	aEntryArray.ResetAndDestroy();
   180 
   181 	// Read contents of install directory
   182 	User::LeaveIfError(dir.Open(iFs, iInstallDir, KEntryAttNormal));
   183     
   184     TEntryArray dirEntries;
   185     TInt readError = KErrNone;
   186     
   187 	while (readError != KErrEof)  
   188 		{
   189 	    readError = dir.Read(dirEntries);
   190     
   191 	    if(readError != KErrNone && readError != KErrEof) 
   192 	    	{
   193 	    	User::Leave(readError);
   194 	    	}
   195 	    else
   196 	    	{
   197 	    	const TInt dirCount = dirEntries.Count();   
   198 	    	for (TInt i=0; i<dirCount; i++)
   199 	    		{
   200 	    		CInstallEntry* installEntry = CInstallEntry::NewL();
   201 	    		CleanupStack::PushL(installEntry);
   202 	    		installEntry->SetL(const_cast<TEntry&>(dirEntries[i]));
   203 	    		if (installEntry->FileExt()==EUnknown)
   204 	    		    {
   205 	    		    CleanupStack::PopAndDestroy();
   206 	    		    }
   207 	    		else
   208 	    		    {
   209 	    		    User::LeaveIfError(aEntryArray.Append(installEntry));
   210 	    		    CleanupStack::Pop(installEntry);
   211 	    		    }
   212 	    		}
   213 	    	}
   214 		}
   215 
   216 	CleanupStack::PopAndDestroy(&dir);
   217 	}
   218 	
   219 TBool CCentRepSWIWatcher::MatchEntries( const CInstallEntry &aSource, const CInstallEntry &aTarget)
   220 	{
   221 	return((aSource.Uid()==aTarget.Uid()) &&
   222 	       (aSource.FileExt()==aTarget.FileExt()));
   223 	}
   224 		
   225 void CCentRepSWIWatcher::FindChangedEntriesL(TBool aStartup)
   226 	{
   227 	// Find added or updated entries
   228 	ReadInstallDirL(iCurrentInstallDirEntries);
   229 		
   230 	// We don't want cache activity during SWI operations
   231 	TServerResources::iCacheManager->DisableCache();
   232 
   233 	TRAPD(err, HandleFileChangesL(aStartup));
   234 	
   235 	// SWI operations finished, enable cache
   236 	TServerResources::iCacheManager->EnableCache();
   237 	User::LeaveIfError(err);
   238 	} 
   239 
   240 void CCentRepSWIWatcher::HandleFileChangesL(TBool aStartup)
   241     {
   242     TInt newCount=iCurrentInstallDirEntries.Count();
   243     TInt currentCount=iInstallEntryArray.Count();
   244     TInt i;
   245     TInt r;
   246     TInt operation;
   247 
   248     // If both counts are 0, we shouldn't have been notified, just return
   249     if( (newCount==0) && (currentCount==0))
   250         return;
   251     
   252     if(aStartup)
   253         {
   254         operation=ESASwisNone;
   255         }
   256     else
   257         {
   258         operation=iSWIOperation;
   259         }
   260 
   261     if( newCount==0)                        // currentCount > 0, newCount = 0
   262         {                                   // All installed files have been deleted
   263         // Handle deletes of all files
   264         for(i=0;i<currentCount;i++)
   265             {
   266             iInstallEntryArray[i]->HandleFileDeleteL(operation);
   267             }
   268         // Free memory of elements
   269         iInstallEntryArray.ResetAndDestroy();
   270         }
   271     else 
   272         {   
   273         if( currentCount==0)                // currentCount = 0, newCount > 0
   274             {                               // All new files need to be handled
   275             for(i=0;i<newCount;i++)
   276                 {
   277                 CInstallEntry* newEntry=iCurrentInstallDirEntries[i];
   278                 newEntry->HandleFileCreateL(operation);
   279                 }
   280             }
   281         else                                // currentCount > 0, newCount > 0
   282             {
   283             // Find added and modified entries by going through new entries and
   284             // looking for them in current array
   285             for(i=0;i<newCount;i++)
   286                 {
   287                 CInstallEntry* newEntry=iCurrentInstallDirEntries[i];
   288                 r=iInstallEntryArray.Find( newEntry, MatchEntries);
   289                 // If we find new entry in current array, check modification date
   290                 if(r>=KErrNone)
   291                     {
   292                     CInstallEntry* currentEntry=iInstallEntryArray[r];
   293                     if( newEntry->Modified() > currentEntry->Modified())
   294                         {                                               
   295                         // Deal with newly installed file, note use newEntry
   296                         // so we use new timestamp
   297                         newEntry->HandleFileUpdateL(operation);
   298                         }
   299                     }
   300                 else if(r==KErrNotFound)        // File has been added
   301                     {
   302                     // Handle add
   303                     newEntry->HandleFileCreateL(operation);
   304                     // Don't leave on KErrNotFound
   305                     r=KErrNone;
   306                     }
   307                 User::LeaveIfError(r);
   308                 }
   309     
   310             // Find deleted entries by going through current entries and looking for them 
   311             // in new array
   312             for(i=0;i<currentCount;i++)
   313                 {
   314                 CInstallEntry* currentEntry=iInstallEntryArray[i];
   315                 r=iCurrentInstallDirEntries.Find( currentEntry, MatchEntries);
   316                 // If we don't find current entry in new array, it's been deleted
   317                 if(r==KErrNotFound)
   318                     {
   319                     // Deal with uninstalls
   320                     currentEntry->HandleFileDeleteL(operation);
   321                     // Don't leave on KErrNotFound
   322                     r=KErrNone;
   323                     }
   324                 User::LeaveIfError(r);
   325                 }
   326             }
   327 
   328         // Clear out old list
   329         iInstallEntryArray.ResetAndDestroy();
   330         
   331         // Re-read directory - if any files were corrupt they have been deleted
   332         // during the merge, so we need to re-read in case this has occurred
   333         ReadInstallDirL(iInstallEntryArray);
   334         }
   335         
   336     SaveInstallDirL();
   337     iCurrentInstallDirEntries.ResetAndDestroy();
   338     }
   339 
   340 CInstallEntry::CInstallEntry() :
   341 	iUid(KNullUid),
   342 	iModified(0),
   343 	iFileExt(EUnknown)
   344 	{
   345 	}
   346 
   347 CInstallEntry::~CInstallEntry()
   348 	{
   349 	if( iRepository)
   350 		{
   351 		iRepository->Close();
   352 		delete iRepository;
   353 		}
   354 	delete iNotifier;
   355 	}
   356 
   357 void CInstallEntry::SetL(TEntry& aEntry)	
   358 	{
   359 	// Get uid from file name
   360 	const TInt KUidLen = 8;
   361 	TPtrC uidPtr=aEntry.iName.Des().LeftTPtr(KUidLen);
   362 	TLex  lex=uidPtr;
   363 
   364 	TUint32 uid;
   365 	User::LeaveIfError(lex.Val(uid,EHex)); 
   366 	iUid.iUid=static_cast<TInt32>(uid);
   367 	
   368 	// save extension type 
   369 	_LIT(KIniFileExtension, ".txt");
   370 	_LIT(KExternalizedFileExt, ".cre");
   371 
   372 	const TInt KExtLen = 4;
   373 	TPtrC extPtr=aEntry.iName.Des().RightTPtr(KExtLen);
   374 	
   375 	if(extPtr.Compare(KIniFileExtension)==0)
   376 		{
   377 		iFileExt=EIni;
   378 		}
   379 	else if(extPtr.Compare(KExternalizedFileExt)==0)
   380 		{
   381 		iFileExt=ECre;
   382 		}
   383 	else
   384 		{
   385 		iFileExt=EUnknown;
   386 		}
   387 		
   388 	iModified=aEntry.iModified;
   389 	}
   390 	
   391 void CInstallEntry::ConstructL()	
   392 	{
   393 	// Create repository object
   394 	iRepository = new(ELeave) CServerRepository;
   395 	// Notifier needed to open repositories.		
   396 	iNotifier = new(ELeave)CSessionNotifier ;
   397 	}
   398 	
   399 CInstallEntry* CInstallEntry::NewL()
   400 	{
   401 	CInstallEntry* self=new(ELeave)CInstallEntry();
   402 	CleanupStack::PushL(self);
   403 	self->ConstructL();
   404 	CleanupStack::Pop(self);
   405 	return self;
   406 	}
   407 
   408 TUid CInstallEntry::Uid() const
   409 	{
   410 	return iUid;
   411 	}
   412 	
   413 TTime CInstallEntry::Modified() const
   414 	{
   415 	return iModified;
   416 	}
   417 	
   418 TCentRepFileType CInstallEntry::FileExt() const
   419 	{
   420 	return iFileExt;
   421 	}
   422  	
   423 void CInstallEntry::ExternalizeL(RWriteStream& aStream) const
   424 	{
   425  	aStream << iUid.iUid;
   426 	aStream << iModified.Int64();
   427 	TUint32 fileExt=iFileExt;
   428 	aStream << fileExt;
   429  	}
   430  	
   431 void CInstallEntry::InternalizeL(RReadStream& aStream)
   432 	{
   433  	aStream >> iUid.iUid;
   434  	TInt64 time;
   435 	aStream >> time;
   436 	iModified=time;
   437 	TUint32 fileExt;
   438 	aStream >> fileExt;
   439 	iFileExt =  static_cast<TCentRepFileType>(fileExt);
   440  	}
   441  		
   442 
   443 void CCentRepSWIWatcher::ReadAndInternalizeInstallDirL(const TDesC& aInstallDirFilePath)
   444 	{
   445 	RFile file;
   446 	User::LeaveIfError(file.Open(TServerResources::iFs,aInstallDirFilePath, EFileRead|EFileShareReadersOnly));
   447 	CleanupClosePushL(file);
   448 
   449 	CDirectFileStore* store = CDirectFileStore::FromLC (file);
   450 	if(store->Type()[0] != KDirectFileStoreLayoutUid)
   451 		{
   452 		User::Leave(KErrCorrupt);
   453 		}
   454 
   455 	iInstallEntryArray.ResetAndDestroy();
   456 
   457 	// Get the root stream and attempt to read the index from it
   458 	TStreamId rootStreamId = store->Root() ;
   459 	RStoreReadStream rootStream ;
   460 	rootStream.OpenLC(*store, rootStreamId);
   461 	
   462 	// Internalize the repository	
   463 	TUint32 count;
   464 	rootStream >> count;
   465 	for(TUint i=0; i<count;i++)
   466 		{
   467 		CInstallEntry* installEntry = CInstallEntry::NewL();
   468 		CleanupStack::PushL(installEntry);
   469 		rootStream >> *installEntry;
   470 		User::LeaveIfError(iInstallEntryArray.Append(installEntry));
   471 		CleanupStack::Pop(installEntry);
   472 		}
   473 		
   474 	CleanupStack::PopAndDestroy(&rootStream);
   475 	CleanupStack::PopAndDestroy(store);
   476 	CleanupStack::PopAndDestroy();
   477 	}
   478 
   479 void CCentRepSWIWatcher::GetInstallDirL()
   480 	{
   481 	_LIT(KInstallDirFile, "installdir.bin");	
   482 	HBufC* filePath = HBufC::NewLC(TServerResources::iDataDirectory->Length() + KInstallDirFile().Length());
   483 	TPtr installDirFilePath(filePath->Des());
   484 	installDirFilePath.Append(*TServerResources::iDataDirectory);
   485 	installDirFilePath.Append(KInstallDirFile);
   486 	
   487 	TRAPD(err, ReadAndInternalizeInstallDirL(installDirFilePath)); // try to open installdir file and internalize its contents
   488 	if (err != KErrNone) 
   489 		{
   490 		TInt fileDeleteErr = TServerResources::iFs.Delete(installDirFilePath);
   491 		// If a debug build - record error
   492 		#ifdef _DEBUG
   493 			if (fileDeleteErr != KErrNone && err != KErrNotFound)
   494 				{
   495 				RDebug::Print(_L("CCentRepSWIWatcher::GetInstallDirL - Failed to delete file. Error = %d"), fileDeleteErr);
   496 				}
   497 		#else
   498 			(void)fileDeleteErr;
   499 		#endif	
   500 			
   501 		// No file saved - read initial contents of directory
   502 		ReadInstallDirL(iInstallEntryArray);
   503 		SaveInstallDirL();	
   504 		}		
   505 	CleanupStack::PopAndDestroy(filePath);		
   506 	}
   507 	
   508 void CCentRepSWIWatcher::SaveInstallDirL() 
   509 	{	
   510 	_LIT(KInstallDirFile, "installdir.bin");
   511 	_LIT(KInstallDirTmpFile, "installdir.tmp");
   512 	
   513 	TBuf<KMaxFileName> installDirTrnsFilePath;
   514 	installDirTrnsFilePath.Append(*TServerResources::iDataDirectory);
   515 	installDirTrnsFilePath.Append(KInstallDirTmpFile);
   516 	
   517 	// Create file store
   518 	CDirectFileStore* store = CDirectFileStore::ReplaceLC(TServerResources::iFs, installDirTrnsFilePath,
   519 		                                                      (EFileWrite | EFileShareExclusive));
   520 	const TUid uid2  = KNullUid ;	                                                     
   521 	store->SetTypeL(TUidType(KDirectFileStoreLayoutUid, uid2, KServerUid3)) ; 
   522 		
   523 	// Write the stream index/dictionary as root stream within the store
   524 	// so we can access it when we do a restore later on
   525 	RStoreWriteStream rootStream ;
   526 	TStreamId rootStreamId = rootStream.CreateLC(*store) ;
   527 	
   528 	TUint32 count=iInstallEntryArray.Count();
   529 	rootStream << count;
   530 	for(TUint i=0; i<count;i++)
   531 		{
   532 		rootStream << *iInstallEntryArray[i];
   533 		}
   534 		
   535 	rootStream.CommitL() ;
   536 		
   537 	CleanupStack::PopAndDestroy(&rootStream) ;
   538 	store->SetRootL(rootStreamId);
   539 	store->CommitL();
   540 	CleanupStack::PopAndDestroy(store) ; 	
   541 	
   542 	TBuf<KMaxFileName> installDirFilePath;
   543 	installDirFilePath.Append(*TServerResources::iDataDirectory);
   544 	installDirFilePath.Append(KInstallDirFile);
   545 
   546     User::LeaveIfError(TServerResources::iFs.Replace(installDirTrnsFilePath, installDirFilePath));	
   547 	}
   548 
   549 
   550 void CInstallEntry::HandleFileDeleteL(TInt aOperation)
   551 	{
   552 	// File should only have been deleted if operation was uninstall
   553 	// or in startup case
   554 	if((aOperation!=ESASwisNone) && (aOperation!=ESASwisUninstall))
   555 		User::Leave(KErrAbort);
   556 					
   557 	iRepository->HandleSWIDeleteL(Uid(), *iNotifier);
   558 	}
   559 	
   560 void CInstallEntry::HandleFileCreateL(TInt aOperation)
   561 	{
   562 	// File should only have been created if operation was install
   563 	// or in startup case
   564 	if((aOperation!=ESASwisNone) && (aOperation!=ESASwisInstall))
   565 		User::Leave(KErrAbort);
   566 					
   567 	iRepository->HandleSWIUpdateL(Uid(), Modified(), *iNotifier);
   568 	}
   569 	
   570 void CInstallEntry::HandleFileUpdateL(TInt aOperation)
   571 	{
   572 	// File should only have been modified if operation was install
   573 	// or in startup case
   574 	if((aOperation!=ESASwisNone) && (aOperation!=ESASwisInstall))
   575 		User::Leave(KErrAbort);
   576 					
   577 	iRepository->HandleSWIUpdateL(Uid(), Modified(), *iNotifier);
   578 	}