First public contribution.
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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Implements installed file notification
23 #include "srvrepos_noc.h"
25 #include "srvparams.h"
28 CCentRepSWIWatcher* CCentRepSWIWatcher::NewL(RFs& aFs)
30 CCentRepSWIWatcher* self = new(ELeave) CCentRepSWIWatcher(aFs);
31 CleanupStack::PushL(self);
33 CleanupStack::Pop(self);
37 void CCentRepSWIWatcher::ConstructL()
39 // Attach to SWI property
40 User::LeaveIfError(iSWIKey.Attach( KUidSystemCategory, KSAUidSoftwareInstallKeyValue));
42 // Initialise SWI operation and status
44 const TInt error = iSWIKey.Get(KUidSystemCategory, KSAUidSoftwareInstallKeyValue, swiProperty);
46 if (error == KErrNone)
48 iSWIOperation=swiProperty & KSASwisOperationMask;
49 iSWIStatus= swiProperty & KSASwisOperationStatusMask;
51 else if (error != KErrNotFound)
53 User::LeaveIfError(error);
56 // Get last saved contents of install directory
59 // Do any actions required by pre server start-up SWI activity
60 TRAP_IGNORE(FindChangedEntriesL(ETrue));
63 CCentRepSWIWatcher::CCentRepSWIWatcher(RFs& aFs) :
64 CActive(EPriorityStandard),
65 iInstallDir(*TServerResources::iInstallDirectory),
67 iSWIOperation( ESASwisNone),
68 iSWIStatus( ESASwisStatusSuccess)
70 CActiveScheduler::Add(this);
73 CCentRepSWIWatcher::~CCentRepSWIWatcher()
76 if(iSWIKey.Handle() != KNullHandle)
81 iInstallEntryArray.ResetAndDestroy();
82 iCurrentInstallDirEntries.ResetAndDestroy();
85 void CCentRepSWIWatcher::Start()
93 void CCentRepSWIWatcher::NotifyChange()
96 // Register for P&S of SWI flag
97 iSWIKey.Subscribe(iStatus);
102 void CCentRepSWIWatcher::RunL()
108 User::LeaveIfError(iSWIKey.Get(KUidSystemCategory, KSAUidSoftwareInstallKeyValue, swiProperty));
110 HandleSWIEventL(swiProperty);
113 void CCentRepSWIWatcher::HandleSWIEventL(TInt aSWIProperty)
115 iSWIOperation=aSWIProperty & KSASwisOperationMask;
116 iSWIStatus= aSWIProperty & KSASwisOperationStatusMask;
118 // Need to handle successful and aborted install/uninstall and successful restore
119 // Can't handle aborted restore
120 switch(iSWIOperation)
125 case ESASwisUninstall:
126 if(iSWIStatus==ESASwisStatusSuccess)
129 FindChangedEntriesL();
131 else if(iSWIStatus==ESASwisStatusAborted)
133 // Update directory to reset timestamps
134 ReadInstallDirL(iInstallEntryArray);
145 // Catch leaves so they don't stop the server
146 TInt CCentRepSWIWatcher::RunError( TInt aError)
148 static_cast<void>(aError);
150 RDebug::Print(_L("Run error %d"), aError);
152 // Reinitialise directory list
153 iInstallEntryArray.ResetAndDestroy();
160 // Renable cache activity in case of errors during SWI events
161 TServerResources::iCacheManager->EnableCache();
166 void CCentRepSWIWatcher::DoCancel()
168 // Cancel subscription to SW P&S flag
173 void CCentRepSWIWatcher::ReadInstallDirL(RPointerArray<CInstallEntry> &aEntryArray)
176 CleanupClosePushL(dir);
178 //Empty contents of directory
179 aEntryArray.ResetAndDestroy();
181 // Read contents of install directory
182 User::LeaveIfError(dir.Open(iFs, iInstallDir, KEntryAttNormal));
184 TEntryArray dirEntries;
185 TInt readError = KErrNone;
187 while (readError != KErrEof)
189 readError = dir.Read(dirEntries);
191 if(readError != KErrNone && readError != KErrEof)
193 User::Leave(readError);
197 const TInt dirCount = dirEntries.Count();
198 for (TInt i=0; i<dirCount; i++)
200 CInstallEntry* installEntry = CInstallEntry::NewL();
201 CleanupStack::PushL(installEntry);
202 installEntry->SetL(const_cast<TEntry&>(dirEntries[i]));
203 if (installEntry->FileExt()==EUnknown)
205 CleanupStack::PopAndDestroy();
209 User::LeaveIfError(aEntryArray.Append(installEntry));
210 CleanupStack::Pop(installEntry);
216 CleanupStack::PopAndDestroy(&dir);
219 TBool CCentRepSWIWatcher::MatchEntries( const CInstallEntry &aSource, const CInstallEntry &aTarget)
221 return((aSource.Uid()==aTarget.Uid()) &&
222 (aSource.FileExt()==aTarget.FileExt()));
225 void CCentRepSWIWatcher::FindChangedEntriesL(TBool aStartup)
227 // Find added or updated entries
228 ReadInstallDirL(iCurrentInstallDirEntries);
230 // We don't want cache activity during SWI operations
231 TServerResources::iCacheManager->DisableCache();
233 TRAPD(err, HandleFileChangesL(aStartup));
235 // SWI operations finished, enable cache
236 TServerResources::iCacheManager->EnableCache();
237 User::LeaveIfError(err);
240 void CCentRepSWIWatcher::HandleFileChangesL(TBool aStartup)
242 TInt newCount=iCurrentInstallDirEntries.Count();
243 TInt currentCount=iInstallEntryArray.Count();
248 // If both counts are 0, we shouldn't have been notified, just return
249 if( (newCount==0) && (currentCount==0))
254 operation=ESASwisNone;
258 operation=iSWIOperation;
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++)
266 iInstallEntryArray[i]->HandleFileDeleteL(operation);
268 // Free memory of elements
269 iInstallEntryArray.ResetAndDestroy();
273 if( currentCount==0) // currentCount = 0, newCount > 0
274 { // All new files need to be handled
275 for(i=0;i<newCount;i++)
277 CInstallEntry* newEntry=iCurrentInstallDirEntries[i];
278 newEntry->HandleFileCreateL(operation);
281 else // currentCount > 0, newCount > 0
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++)
287 CInstallEntry* newEntry=iCurrentInstallDirEntries[i];
288 r=iInstallEntryArray.Find( newEntry, MatchEntries);
289 // If we find new entry in current array, check modification date
292 CInstallEntry* currentEntry=iInstallEntryArray[r];
293 if( newEntry->Modified() > currentEntry->Modified())
295 // Deal with newly installed file, note use newEntry
296 // so we use new timestamp
297 newEntry->HandleFileUpdateL(operation);
300 else if(r==KErrNotFound) // File has been added
303 newEntry->HandleFileCreateL(operation);
304 // Don't leave on KErrNotFound
307 User::LeaveIfError(r);
310 // Find deleted entries by going through current entries and looking for them
312 for(i=0;i<currentCount;i++)
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
319 // Deal with uninstalls
320 currentEntry->HandleFileDeleteL(operation);
321 // Don't leave on KErrNotFound
324 User::LeaveIfError(r);
328 // Clear out old list
329 iInstallEntryArray.ResetAndDestroy();
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);
337 iCurrentInstallDirEntries.ResetAndDestroy();
340 CInstallEntry::CInstallEntry() :
347 CInstallEntry::~CInstallEntry()
351 iRepository->Close();
357 void CInstallEntry::SetL(TEntry& aEntry)
359 // Get uid from file name
360 const TInt KUidLen = 8;
361 TPtrC uidPtr=aEntry.iName.Des().LeftTPtr(KUidLen);
365 User::LeaveIfError(lex.Val(uid,EHex));
366 iUid.iUid=static_cast<TInt32>(uid);
368 // save extension type
369 _LIT(KIniFileExtension, ".txt");
370 _LIT(KExternalizedFileExt, ".cre");
372 const TInt KExtLen = 4;
373 TPtrC extPtr=aEntry.iName.Des().RightTPtr(KExtLen);
375 if(extPtr.Compare(KIniFileExtension)==0)
379 else if(extPtr.Compare(KExternalizedFileExt)==0)
388 iModified=aEntry.iModified;
391 void CInstallEntry::ConstructL()
393 // Create repository object
394 iRepository = new(ELeave) CServerRepository;
395 // Notifier needed to open repositories.
396 iNotifier = new(ELeave)CSessionNotifier ;
399 CInstallEntry* CInstallEntry::NewL()
401 CInstallEntry* self=new(ELeave)CInstallEntry();
402 CleanupStack::PushL(self);
404 CleanupStack::Pop(self);
408 TUid CInstallEntry::Uid() const
413 TTime CInstallEntry::Modified() const
418 TCentRepFileType CInstallEntry::FileExt() const
423 void CInstallEntry::ExternalizeL(RWriteStream& aStream) const
425 aStream << iUid.iUid;
426 aStream << iModified.Int64();
427 TUint32 fileExt=iFileExt;
431 void CInstallEntry::InternalizeL(RReadStream& aStream)
433 aStream >> iUid.iUid;
439 iFileExt = static_cast<TCentRepFileType>(fileExt);
443 void CCentRepSWIWatcher::ReadAndInternalizeInstallDirL(const TDesC& aInstallDirFilePath)
446 User::LeaveIfError(file.Open(TServerResources::iFs,aInstallDirFilePath, EFileRead|EFileShareReadersOnly));
447 CleanupClosePushL(file);
449 CDirectFileStore* store = CDirectFileStore::FromLC (file);
450 if(store->Type()[0] != KDirectFileStoreLayoutUid)
452 User::Leave(KErrCorrupt);
455 iInstallEntryArray.ResetAndDestroy();
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);
462 // Internalize the repository
465 for(TUint i=0; i<count;i++)
467 CInstallEntry* installEntry = CInstallEntry::NewL();
468 CleanupStack::PushL(installEntry);
469 rootStream >> *installEntry;
470 User::LeaveIfError(iInstallEntryArray.Append(installEntry));
471 CleanupStack::Pop(installEntry);
474 CleanupStack::PopAndDestroy(&rootStream);
475 CleanupStack::PopAndDestroy(store);
476 CleanupStack::PopAndDestroy();
479 void CCentRepSWIWatcher::GetInstallDirL()
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);
487 TRAPD(err, ReadAndInternalizeInstallDirL(installDirFilePath)); // try to open installdir file and internalize its contents
490 TInt fileDeleteErr = TServerResources::iFs.Delete(installDirFilePath);
491 // If a debug build - record error
493 if (fileDeleteErr != KErrNone && err != KErrNotFound)
495 RDebug::Print(_L("CCentRepSWIWatcher::GetInstallDirL - Failed to delete file. Error = %d"), fileDeleteErr);
501 // No file saved - read initial contents of directory
502 ReadInstallDirL(iInstallEntryArray);
505 CleanupStack::PopAndDestroy(filePath);
508 void CCentRepSWIWatcher::SaveInstallDirL()
510 _LIT(KInstallDirFile, "installdir.bin");
511 _LIT(KInstallDirTmpFile, "installdir.tmp");
513 TBuf<KMaxFileName> installDirTrnsFilePath;
514 installDirTrnsFilePath.Append(*TServerResources::iDataDirectory);
515 installDirTrnsFilePath.Append(KInstallDirTmpFile);
518 CDirectFileStore* store = CDirectFileStore::ReplaceLC(TServerResources::iFs, installDirTrnsFilePath,
519 (EFileWrite | EFileShareExclusive));
520 const TUid uid2 = KNullUid ;
521 store->SetTypeL(TUidType(KDirectFileStoreLayoutUid, uid2, KServerUid3)) ;
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) ;
528 TUint32 count=iInstallEntryArray.Count();
530 for(TUint i=0; i<count;i++)
532 rootStream << *iInstallEntryArray[i];
535 rootStream.CommitL() ;
537 CleanupStack::PopAndDestroy(&rootStream) ;
538 store->SetRootL(rootStreamId);
540 CleanupStack::PopAndDestroy(store) ;
542 TBuf<KMaxFileName> installDirFilePath;
543 installDirFilePath.Append(*TServerResources::iDataDirectory);
544 installDirFilePath.Append(KInstallDirFile);
546 User::LeaveIfError(TServerResources::iFs.Replace(installDirTrnsFilePath, installDirFilePath));
550 void CInstallEntry::HandleFileDeleteL(TInt aOperation)
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);
557 iRepository->HandleSWIDeleteL(Uid(), *iNotifier);
560 void CInstallEntry::HandleFileCreateL(TInt aOperation)
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);
567 iRepository->HandleSWIUpdateL(Uid(), Modified(), *iNotifier);
570 void CInstallEntry::HandleFileUpdateL(TInt aOperation)
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);
577 iRepository->HandleSWIUpdateL(Uid(), Modified(), *iNotifier);